我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Java如何使用Socket正确读取数据姿势

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Java如何使用Socket正确读取数据姿势

小编给大家分享一下Java如何使用Socket正确读取数据姿势,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!

    前言

    平时日常开发用得最多是Http通讯,接口调试也比较简单的,也有比较强大的框架支持(OkHttp)。

    个人平时用到socket通讯的地方是Android与外设通讯,Android与ssl服务通讯,这种都是基于TCP/IP通讯,而且服务端和设备端协议都是不能修改的,只能按照相关报文格式进行通信。

    但使用socket通讯问题不少,一般有两个难点:

    socket通讯层要自己写及IO流不正确使用,遇到读取不到数据或者阻塞卡死现象或者数据读取不完整

    请求和响应报文格式多变(json,xml,其它),解析麻烦,如果是前面两种格式都简单,有对应框架处理,其它格式一般都需要自己手动处理。

    本次基于第1点问题做了总结,归根结底是使用read()或readLine()导致的问题

    Socket使用流程

    创建socket

    连接socket

    获取输入输出流

    字节流:

       InputStream  mInputStream = mSocket.getInputStream();   OutputStream  mOutputStream = mSocket.getOutputStream();

    字符流:

      BufferedReader mBufferedReader = new BufferedReader(new InputStreamReader(mSocket.getInputStream(), "UTF-8"));  PrintWriter mPrintWriter = new PrintWriter(new BufferedWriter(new OutputStreamWriter(mSocket.getOutputStream(), "UTF-8")), true);

    至于实际使用字节流还是字符流,看实际情况使用。如果返回是字符串及读写与报文结束符(/r或/n或/r/n)有关,使用字符流读取,否则字节流。

    读写数据

    关闭socket

    如果是Socket短连接,上面五个步骤都要走一遍;

    如果是Socket长连接,只需关注第4点即可,第4点使用不慎就会遇到上面出现的问题。

    实际开发中,长连接使用居多,一次连接,进行多次收发数据。

    特别注意:使用长连接不能读完数据后立马关闭输入输出流,必须再最后不使用的时候关闭

    Socket数据读写

    当socket阻塞时,必须设置读取超时时间,防止调试时,socket读取数据长期挂起。

    mSocket.setSoTimeout(10* 1000);  //设置客户端读取服务器数据超时时间

    使用read()读取阻塞问题

    日常写法1:

     mOutputStream.write(bytes); mOutputStream.flush();byte[] buffer = new byte[1024];int n = 0;ByteArrayOutputStream output = new ByteArrayOutputStream();while (-1 != (n = mInputStream .read(buffer))) {    output.write(buffer, 0, n);}//处理数据  output.close();byte[] result = output.toByteArray();

    上面看似没有什么问题,但有时候会出现mInputStream .read(buffer)阻塞,导致while循环体里面不会执行

    日常写法2:

    mOutputStream.write(bytes);mOutputStream.flush();int  available = mInputStream.available();byte[] buffer = new byte[available];in.read(buffer);

    上面虽然不阻塞,但不一定能读取到数据,available 可能为0,由于是网络通讯,发送数据后不一定马上返回。

    或者对mInputStream.available()修改为:

     int available = 0;while (available == 0) {    available = mInputStream.available();}

    上面虽然能读取到数据,但数据不一定完整。

    而且,available方法返回估计的当前流可用长度,不是当前通讯流的总长度,而且是估计值;read方法读取流中数据到buffer中,但读取长度为1至buffer.length,若流结束或遇到异常则返回-1。

    最终写法(递归读取):

         public void readStreamWithRecursion(ByteArrayOutputStream output, InputStream inStream) throws Exception {        long start = System.currentTimeMillis();        while (inStream.available() == 0) {            if ((System.currentTimeMillis() - start) > 20* 1000) {//超时退出                throw new SocketTimeoutException("超时读取");            }        }        byte[] buffer = new byte[2048];        int read = inStream.read(buffer);        output.write(buffer, 0, read);        SystemClock.sleep(100);//需要延时以下,不然还是有概率漏读        int a = inStream.available();//再判断一下,是否有可用字节数或者根据实际情况验证报文完整性        if (a > 0) {            LogUtils.w("========还有剩余:" + a + "个字节数据没读");            readStreamWithRecursion(output, inStream);        }    }        private byte[] readStream(InputStream inStream) throws Exception {        ByteArrayOutputStream output = new ByteArrayOutputStream();        readStreamWithRecursion(output, inStream);        output.close();        int size = output.size();        LogUtils.i("本次读取字节总数:" + size);        return output.toByteArray();    }

    上面这种方法读取完成一次后,固定等待时间,等待完不一定有数据,若没有有数据,响应时间过长,会影响用户体验。我们可以再优化一下:

         public void readStreamWithRecursion(ByteArrayOutputStream output, InputStream inStream) throws Exception {        long start = System.currentTimeMillis();        int time =500;//毫秒,间看实际情况        while (inStream.available() == 0) {            if ((System.currentTimeMillis() - start) >time) {//超时退出                throw new SocketTimeoutException("超时读取");            }        }        byte[] buffer = new byte[2048];        int read = inStream.read(buffer);        output.write(buffer, 0, read);       int wait = readWait();        long startWait = System.currentTimeMillis();        boolean checkExist = false;        while (System.currentTimeMillis() - startWait <= wait) {            int a = inStream.available();            if (a > 0) {                checkExist = true;                //            LogUtils.w("========还有剩余:" + a + "个字节数据没读");                break;            }        }        if (checkExist) {            if (!checkMessage(buffer, read)) {                readStreamWithRecursion(output, inStream, timeout);            }        }            }         protected int readWait() {        return 100;    }            private byte[] readStream(InputStream inStream) throws Exception {        ByteArrayOutputStream output = new ByteArrayOutputStream();        readStreamWithRecursion(output, inStream);        output.close();        int size = output.size();        LogUtils.i("本次读取字节总数:" + size);        return output.toByteArray();    }

    上面这种延迟率大幅降低,目前正在使用该方法读取,再也没有出现数据读取不完整和阻塞现象。不过这种,读取也要注意报文结束符问题,何时读取完毕问题。

    使用readreadLine()读取阻塞问题

    日常写法:

     mPrintWriter.print(sendData+ "\r\n");    mPrintWriter.flush(); String msg = mBufferedReader.readLine(); //处理数据

    细心的你发现,发送数据时添加了结束符,如果不加结束符,导致readLine()阻塞,读不到任何数据,最终抛出SocketTimeoutException异常

    特别注意:

    报文结束符:根据实际服务器规定的来添加,必要时问后端开发人员或者看接口文档是否有说明

    不然在接口调试上会浪费很多宝贵的时间,影响后期功能开发。

    使用readLine()注意事项:

    • 读入的数据要注意有/r或/n或/r/n

    这句话意思是服务端写完数据后,会打印报文结束符/r或/n或/r/n;

    同理,客户端写数据时也要打印报文结束符,这样服务端才能读取到数据。

    • 没有数据时会阻塞,在数据流异常或断开时才会返回null

    • 使用socket之类的数据流时,要避免使用readLine(),以免为了等待一个换行/回车符而一直阻塞

    上面长连接是发送一次数据和读一次数据,保证了当次通讯的完整性,必须要时需要同步处理。

    也有长连接,客户端开线程循环阻塞等待服务端数据发送数据过来,比如:消息推送。平时使用长连接都是分别使用不同的命令发送数据且接收数据,来完成不同的任务。

    看完了这篇文章,相信你对“Java如何使用Socket正确读取数据姿势”有了一定的了解,如果想了解更多相关知识,欢迎关注编程网行业资讯频道,感谢各位的阅读!

    免责声明:

    ① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

    ② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

    Java如何使用Socket正确读取数据姿势

    下载Word文档到电脑,方便收藏和打印~

    下载Word文档

    猜你喜欢

    Java如何使用Socket正确读取数据姿势

    小编给大家分享一下Java如何使用Socket正确读取数据姿势,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!前言平时日常开发用得最多是Http通讯,接口调试也比较简单的,也有比较强大的框架支持(OkHttp)。个人平时用到
    2023-06-25

    如何使用pandas正确读取txt文件

    如何使用pandas正确读取txt文件,需要具体代码示例Pandas是一个广泛使用的Python数据分析库,它可以用于处理各种各样的数据类型,包括CSV文件、Excel文件、SQL数据库等。同时,它也可以用于读取文本文件,例如txt文件。
    如何使用pandas正确读取txt文件
    2024-01-19

    如何在C#中使用Socket读取超大的数据

    这期内容当中小编将会给大家带来有关如何在C#中使用Socket读取超大的数据,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。/// /// 返回摄像头信息 /
    2023-06-06

    如何使用ADO.NET读取数据

    小编给大家分享一下如何使用ADO.NET读取数据,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!ADO.NET还是比较常用的,于是我研究了一下ADO.NET读取数据
    2023-06-17

    Go语言如何利用Mutex保障数据读写正确

    这篇文章主要介绍了互斥锁的实现机制,以及 Go 标准库的互斥锁 Mutex 的基本使用方法,文中的示例代码讲解详细,需要的小伙伴可以参考一下
    2023-05-20

    go gin如何正确读取http response body内容并多次使用

    这篇文章主要介绍了go gin如何正确读取http response body内容并多次使用的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇go gin如何正确读取http response body内容并多次使
    2023-07-04

    如何正确使用rowcount函数进行数据统计

    如何正确使用rowcount函数进行数据统计,需要具体代码示例在进行数据统计时,我们经常会使用到SQL语句来对数据库中的数据进行查询和分析。而在某些情况下,我们需要统计查询结果的行数,以便进行进一步的数据处理和分析。这时,就可以借助数据库提
    如何正确使用rowcount函数进行数据统计
    2023-12-29

    如何使用vbscript读取Access数据库

    这篇文章主要介绍如何使用vbscript读取Access数据库,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!效果如图所示:核心代码:代码如下:Option Explicit Dim arrTables( ), i,
    2023-06-08

    Java如何使用jxl读取excel并保存到数据库

    小编给大家分享一下Java如何使用jxl读取excel并保存到数据库,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!项目中涉及到读取excel中的数据,保存到数据库
    2023-05-31

    如何在Python中使用pandas读取数据

    今天就跟大家聊聊有关如何在Python中使用pandas读取数据,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。一、三种数据文件的读取二、csv、tsv、txt 文件读取1)CSV文件
    2023-06-15

    编程热搜

    • Python 学习之路 - Python
      一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
      Python 学习之路 - Python
    • chatgpt的中文全称是什么
      chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
      chatgpt的中文全称是什么
    • C/C++中extern函数使用详解
    • C/C++可变参数的使用
      可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
      C/C++可变参数的使用
    • css样式文件该放在哪里
    • php中数组下标必须是连续的吗
    • Python 3 教程
      Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
      Python 3 教程
    • Python pip包管理
      一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
      Python pip包管理
    • ubuntu如何重新编译内核
    • 改善Java代码之慎用java动态编译

    目录