Java Socket正确读取数据姿势

2024-05-29 23:48

本文主要是介绍Java Socket正确读取数据姿势,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言

平时日常开发用得最多是Http通讯,接口调试也比较简单的,也有比较强大的框架支持(OkHttp)。
个人平时用到socket通讯的地方是Android与外设通讯,Android与ssl服务通讯,这种都是基于TCP/IP通讯,而且服务端和设备端协议都是不能修改的,只能按照相关报文格式进行通信。
但使用socket通讯问题不少,一般有两个难点:

  1. socket通讯层要自己写及IO流不正确使用,遇到读取不到数据或者阻塞卡死现象或者数据读取不完整
  2. 请求和响应报文格式多变(json,xml,其它),解析麻烦,如果是前面两种格式都简单,有对应框架处理,其它格式一般都需要自己手动处理。

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

Socket使用流程

  1. 创建socket
  2. 连接socket
  3. 获取输入输出流

字节流:

   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(0x0A)或/n(0x0D)或/r/n)有关,使用字符流读取,否则字节流。

  1. 读写数据
  2. 关闭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。

最终写法(递归读取):

 /*** 递归读取流** @param output* @param inStream* @param timeout  单位毫秒,看实际情况定义(如果socket通讯超时时间略长,如果用于硬件串口数据读取则略短),可以指定200毫秒,10*1000或20*1000* @return* @throws Exception*/public void readStreamWithRecursion(ByteArrayOutputStream output, InputStream inStream, int timeout) throws Exception {long start = System.currentTimeMillis();while (inStream.available() == 0) {if ((System.currentTimeMillis() - start) > timeout) {//超时退出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,timeout);}}/*** 读取字节** @param inStream* @param timeout* @return* @throws Exception*/private byte[] readStream(InputStream inStream, int timeout) throws Exception {ByteArrayOutputStream output = new ByteArrayOutputStream();readStreamWithRecursion(output, inStream,timeout);output.close();int size = output.size();LogUtils.i("本次读取字节总数:" + size);return output.toByteArray();}

上面流读取的时候给定一个读超时时间,并在读取完成一次后,固定等待时间,等待完不一定有数据,若没有有数据,响应时间过长,会影响用户体验。我们可以再优化一下:

 /*** 递归读取流** @param output* @param inStream* @param timeout* @return* @throws Exception*/public void readStreamWithRecursion(ByteArrayOutputStream output, InputStream inStream, int timeout) throws Exception {long start = System.currentTimeMillis();while (inStream.available() == 0) {if ((System.currentTimeMillis() - start) >timeout) {//超时退出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) {readStreamWithRecursion(output, inStream, timeout);}}/*** 二次读取最大等待时间,单位毫秒*/protected int readWait() {return 100;}/*** 读取字节** @param inStream* @param timeout* @return* @throws Exception*/private byte[] readStream(InputStream inStream, int timeout) throws Exception {ByteArrayOutputStream output = new ByteArrayOutputStream();readStreamWithRecursion(output, inStream,timeout);output.close();int size = output.size();LogUtils.i("本次读取字节总数:" + size);return output.toByteArray();}

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

细心的朋友可以发现上面适合于客服端主动向服务端发起请求响应处理(一问一答),不适合做监听服务端数据处理,监听还得轮询阻塞方式处理,可以参考文章最后的链接。

上面字节流readStreamWithRecursion方法使用备注:

  • 如果是socket通讯建议设置mSocket.setSoTimeout读超时时间,readStreamWithRecursion方法读取超时时间小于SoTimeout
  • 如果是用于硬件通讯流数据读取,非常合适,可以对发送的任意指令设置读取超时间时间

补充

如果你实际Socket通讯数据包格式是有一定规则的,可以把InputStream、OutputStream 分别装饰成 DataOutputStreamDataOutputStream

public DataInputStream(InputStream in) { super(in);}
public DataOutputStream(OutputStream out) { super(out); }

这样更方便读写数据,边读边解析数据,转成相应的基本Java数据类型。这与上面使用read()一次性读取数据到ByteArrayOutputStream ,然后根据byte[]提取相应的数据。不过这两种方式各有优缺点,看具体情况使用。

关于这两个DataOutputStreamDataOutputStream的具体用法参考文章全网最全的Java Socket通讯例子

使用readreadLine()读取阻塞问题

日常写法:

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

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

特别注意:
报文结束符:根据实际服务器规定的来添加,必要时问后端开发人员或者看接口文档是否有说明
不然在接口调试上会浪费很多宝贵的时间,影响后期功能开发。

使用readLine()注意事项:

  1. 读入的数据要注意有/r或/n或/r/n
    这句话意思是服务端写完数据后,会打印报文结束符/r或/n或/r/n;
    同理,客户端写数据时也要打印报文结束符,这样服务端才能读取到数据。
  2. 没有数据时会阻塞,在数据流异常或断开时才会返回null
  3. 使用socket之类的数据流时,要避免使用readLine(),以免为了等待一个换行/回车符而一直阻塞

上面长连接是发送一次数据和读一次数据,保证了当次通讯的完整性,必须要时需要同步处理。
也有长连接,客户端开线程循环阻塞等待服务端数据发送数据过来,比如:消息推送。平时使用长连接都是分别使用不同的命令发送数据且接收数据,来完成不同的任务。

总结

实际开发中,长连接比较复杂,还要考虑心跳,丢包,断开重连等问题。使用长连接时,要特别注意报文结束符问题,结束符只是用来告诉客户端或服务端数据已经发送完毕,客户端或服务端可以读取数据了,否则客户端或服务端会一直阻塞在read()或者readLine()方法。

其它文章:

全网最全的Java Socket通讯例子

这篇关于Java Socket正确读取数据姿势的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1014980

相关文章

Java实现删除文件中的指定内容

《Java实现删除文件中的指定内容》在日常开发中,经常需要对文本文件进行批量处理,其中,删除文件中指定内容是最常见的需求之一,下面我们就来看看如何使用java实现删除文件中的指定内容吧... 目录1. 项目背景详细介绍2. 项目需求详细介绍2.1 功能需求2.2 非功能需求3. 相关技术详细介绍3.1 Ja

springboot项目中整合高德地图的实践

《springboot项目中整合高德地图的实践》:本文主要介绍springboot项目中整合高德地图的实践,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一:高德开放平台的使用二:创建数据库(我是用的是mysql)三:Springboot所需的依赖(根据你的需求再

spring中的ImportSelector接口示例详解

《spring中的ImportSelector接口示例详解》Spring的ImportSelector接口用于动态选择配置类,实现条件化和模块化配置,关键方法selectImports根据注解信息返回... 目录一、核心作用二、关键方法三、扩展功能四、使用示例五、工作原理六、应用场景七、自定义实现Impor

SpringBoot3应用中集成和使用Spring Retry的实践记录

《SpringBoot3应用中集成和使用SpringRetry的实践记录》SpringRetry为SpringBoot3提供重试机制,支持注解和编程式两种方式,可配置重试策略与监听器,适用于临时性故... 目录1. 简介2. 环境准备3. 使用方式3.1 注解方式 基础使用自定义重试策略失败恢复机制注意事项

SpringBoot整合Flowable实现工作流的详细流程

《SpringBoot整合Flowable实现工作流的详细流程》Flowable是一个使用Java编写的轻量级业务流程引擎,Flowable流程引擎可用于部署BPMN2.0流程定义,创建这些流程定义的... 目录1、流程引擎介绍2、创建项目3、画流程图4、开发接口4.1 Java 类梳理4.2 查看流程图4

一文详解如何在idea中快速搭建一个Spring Boot项目

《一文详解如何在idea中快速搭建一个SpringBoot项目》IntelliJIDEA作为Java开发者的‌首选IDE‌,深度集成SpringBoot支持,可一键生成项目骨架、智能配置依赖,这篇文... 目录前言1、创建项目名称2、勾选需要的依赖3、在setting中检查maven4、编写数据源5、开启热

Java对异常的认识与异常的处理小结

《Java对异常的认识与异常的处理小结》Java程序在运行时可能出现的错误或非正常情况称为异常,下面给大家介绍Java对异常的认识与异常的处理,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参... 目录一、认识异常与异常类型。二、异常的处理三、总结 一、认识异常与异常类型。(1)简单定义-什么是

SpringBoot项目配置logback-spring.xml屏蔽特定路径的日志

《SpringBoot项目配置logback-spring.xml屏蔽特定路径的日志》在SpringBoot项目中,使用logback-spring.xml配置屏蔽特定路径的日志有两种常用方式,文中的... 目录方案一:基础配置(直接关闭目标路径日志)方案二:结合 Spring Profile 按环境屏蔽关

Java使用HttpClient实现图片下载与本地保存功能

《Java使用HttpClient实现图片下载与本地保存功能》在当今数字化时代,网络资源的获取与处理已成为软件开发中的常见需求,其中,图片作为网络上最常见的资源之一,其下载与保存功能在许多应用场景中都... 目录引言一、Apache HttpClient简介二、技术栈与环境准备三、实现图片下载与保存功能1.

SpringBoot排查和解决JSON解析错误(400 Bad Request)的方法

《SpringBoot排查和解决JSON解析错误(400BadRequest)的方法》在开发SpringBootRESTfulAPI时,客户端与服务端的数据交互通常使用JSON格式,然而,JSON... 目录问题背景1. 问题描述2. 错误分析解决方案1. 手动重新输入jsON2. 使用工具清理JSON3.