Connection reset问题的分析和解决
问题情景
有一个服务端,连接了多个客户端以数组存储管理,服务端开启了一个线程进行文件传输操作。这样服务端能够正确连接到每一个客户端。但是他尝试向每个客户端分别发送代码的时候却出现了服务端发送的消息只能发送给第一个获取到输出流的客户端然后报出“Connection reset”的错误
问题代码
public void creat() {
try {
ServerSocket ssoc = new ServerSocket(16666);
System.out.println("服务器已启动等待客户端连接。。。");
int count = 0;
for(int i = 0;i < 3;i++) {
System.out.println("这是第" + i + "次连接");
Socket soc = ssoc.accept();
socList[count] = soc;
System.out.println("客户端" + soc.getPort() + "已连接");
count++;
}
for(int i = 0;i < socList.length;i++) {
Thread.sleep(500);
OutputStream op = socList[i].getOutputStream();
String str = new String("hello" + socList[i].getPort() );
System.out.println("发送消息给" + socList[i].getPort());
op.write(str.getBytes());
}
} catch (IOException e) {
e.printStackTrace();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
如代码所示我们连接到了3个客户端并且也分别获取到了他们对应的输出流向他们发送输出消息
如图所示
而客户端呢?
可见只有第一次连接的“57715客户端”收到了一次服务端发来的hello而我们其他客户端却没有接收到hello并且报出了“Connection reset”的错误?这是怎么回事呢从代码明面上我们并没有出错啊?
问题分析
客户端和服务端之间是怎么连接的
首先我们这里的客户端和服务端是基于TCP协议进行的数据传输
TCP协议的具体内容这里就不再赘述,需要的同学可以区看看相关《计算机网络》的书籍
简单来说TCP协议也就是基于某种特定的数据发送接收方式进行的端对端的数据传输的一个协定
需要在两个端口“匹配”且“连接成功”的情况下遵守这个协议下数据传输的要求进行数据传输,
就好比你打电话给某个人,你首先需要知道它的电话号码,并且你需要拨号并且在打通电话的情况下才能和他通话,如果你打过去他占线了的话你也不能通话
而在建立起两个端口连接前需要经历三次握手,而两个端口断开需要四次挥手
首先我们来看个图
TCP协议下要建立两个端口的连接就需要进行三次握手只有三次握手都确认的情况下才算双方ESTABLISHED建立成功
第一次握手:首先Client向Server发送 SYN信息请求建立连接
? Server监听到这个信息之后同意连接的话就会给Client发送(SYN+ACK)的字报段进入SYN_RCVD状态
第二次握手:Client收到Server同意的信息之后就确认ESTABLISHED建立了连接然后发送ACK消息给Server
第三次握手:Server收到Client的ACK确认建立消息后那么Server也就确认ESTABLISHED连接建立成功
为什么要用三次握手呢?为什么不用两次握手呢?
因为我们会遇到Client向Server发送请求建立连接的消息,而这个消息已经超时了,当Server接收到消息的时候
Client已经没有建立连接的想法 了,如果没有三次握手的确认,当Server单方面确认建立连接的话那么这个连接就确认成功了
Server会一直等待Client发送消息来而Client完全没有已经到连接建立而不会有数据传过去,这个就会导致连接资源的浪费
其实基本的思路和三次握手是一样的
第一次挥手:当主机1调用close()方法的时候表明它已经无法再进行数据的传输和接收了,它会发送一个FIN信息给主机2
第二次挥手:当主机2收到FIN信息的时候它知道了主机1想要关闭连接通道主机2就会发送确认消息ACK让主机1知道自己确认了这个 事情但是也许还有未完的数据没有发送
第三次挥手:收到主机1的FIN消息之后它如果没有数据要传输了的话就会调用close()方法关闭连接然后发送FIN消息给主机1让它知 道自己也关闭连接不会再发送和接收消息
第四次挥手:收到主机2发送的FIN信息之后主机1明白了主机2连接通道也关闭了所以会发送一个确认ACK消息给主机2作为最后的分 别让象征着这一次连接成功断开
为什么要用四次分手呢?
因为我们无法保证某一方想要断开连接 的时候双方的文件传输是否已经结束,所以我们需要进行双方的一个ACK信息的一个确立,如上述主机1和主机2,如果主机2接收到FIN信息之后还需要发送消息给主机1的话那么它就会发送完消息之后再调用close()方法给主机1发送FIN信息,而主机1只有收到主机2的FIN信息之后才会给主机2发送ACK信息告知这段连接的结束表面他们已经“正式分手”了
三次握手和四次挥手的图片和部分内容摘自此博客:(70条消息) 什么是TCP协议?_YUAN的博客-CSDN博客_tcp协议有兴趣的朋友可以看看
什么是Connection reset?
首先我们来看看正常断开和异常强制断开的区别,这个是我看Oracle网站上一篇Connection release的文章的总结连接附后。
JAVA中A、B两个端口基于TCP协议下有序(orderly)断开或者强制断开(abortive)的区别
- 在TCP中两个端口是分别单独(separate)
- 但是是传输数据的流(stream)半独立(semi-independent)的
有序(orderly)
- 如果有一方输送数据的流关闭(close)了,那么会向TCP栈发送”FIN“信号,当TCP收到"FIN"信号之后另一方无论是否读完消息都会收到-1的返回值然后停止读取数据并且执行关闭流的那方的同样操作发送“FIN”信号A端也停止接受信号
- “FIN”信息标志着这一端口已经完成了”发送消息“不会再发送数据了
- Socket.close();标志着我不仅完成了发送也完成了接收(receive)
流产(abortive)
- abortive关闭使用的是"RST"信息来终止连接,只要双方中某一方有一个TCP栈发送“RST”信息的异常情况,那么TCP栈中的数据全部都会因为不在被接收或者发送而丢失
- 一个从BSD sockets创建就存在的一个处理方法" ‘linger’ sockets release option",它用于强制执行abortive连接释放。
- 任意一方可以给TCP传输Socket.setLinger(true,0),当它将要关闭流的时候,这个操作不会立即起作用直到它随后调用Socket.close();
- 然后这个连接就会被发送的”RST“消息给“流产”
- 当然还存在其他情况使这个连接流产但这个方式是最简单的
他们之间的区别
- close()方法在两个模型中都有使用,但是区别就是在orderly模型中双方都使用了Socket.close()的方法而在abortive模型中只有一方使用了close()方法
- 且abortive模型设置了linger(0)的操作,以“RST”信息去提示对方连接中断。而orderly模型中使用的是”FIN“信息去提醒另一端调用close()方法而关闭流
流产(中断)连接模型实例
因为在调用close方法之后这一个端口不再会进行消息的发送和接收但是你不发送消息了,可能还会有另一端来的消息传输过来需要接受。此时就会是第一种模型
当A尝试使用orderly(有序地)关闭连接的时候,B还在不停的发送消息。这个时候已经只有一方连接而另一端关闭了。这个时候没有收到响应的B一定会不停发送消息,所以这个时候A的TCP栈必须要发送”RST“消息来强制终止这个连接
- 一方给另一方发消息但是另一方在没有接收任何消息的时候关闭了
这个时候接收消息的一方意识到了有数据丢失,所以相比之下使用”FIN“来使发送消息一方正常关闭,使用”RST“消息让A得到一个SocketExecption报错会更加正确
原文连接:https://docs.oracle.com/javase/1.5.0/docs/guide/net/articles/connection_release.html
问题实例分析
所以我们上述将的orderly式的关闭其实也就是前面所说的“四次挥手模型”的关闭,
而我们遇到的问题也就是出在了四次挥手的途中,有一方根本没有发送FIN消息给对方并且一直发送消息
所以变成了abortive式的连接断开。
这里是哪种情况呢
for(int i = 0;i < socList.length;i++) {
Thread.sleep(500);
OutputStream op = socList[i].getOutputStream();
String str = new String("hello" + socList[i].getPort() );
System.out.println("发送消息给" + socList[i].getPort());
op.write(str.getBytes());
}
- 从我们最开始的代码来看我们每一次获取完OutputStream对象之后发送完消息,
? 都没有调用close()方法,所以客户端也一直没有收到FIN信息,从而一直保持连接状态,
? 另一端接入的是上一个客户端的输入流,连接无法正常建立,所以即使你的数据是正常传输出去的
? 但是对应客户端的输入流read()的时候因为没有正确连接而导致收到“RST”信息而强行中断进程。
- 这就像是你打电话给对面,对面正忙你无法正确地和他通话一样。
解决方法
每次你使用完某个输出流之后都将相应的输出流关闭,
相应的输入流会受到FIN信息从而自动关闭它的输入流。
也就是你只有一个手机和这个人通完话记得点击挂断否则下一个人电话打不进来了。
for(int i = 0;i < socList.length;i++) {
Thread.sleep(500);
OutputStream op = socList[i].getOutputStream();
String str = new String("hello" + socList[i].getPort() );
System.out.println("发送消息给" + socList[i].getPort());
op.write(str.getBytes());
op.close();
}
这样就能释放IO流的连接资源,保障每次连接都正确成功。
ps:这一次文章是自己第一次做出这样的尝试进行问题分析,有不对的地方希望各位大佬即使指出
|