IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 网络协议 -> Connection reset问题的分析和解决 -> 正文阅读

[网络协议]Connection reset问题的分析和解决

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) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (InterruptedException e) {
			// TODO Auto-generated catch block
			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()方法而关闭流

流产(中断)连接模型实例

  • 一方关闭尝试用”FIN“终止另一方而另一端还在不停的发送消息

    这个会返回”Connection Release by peer“(如果是不停的在读信息那么就会返回”Connection Release“)

因为在调用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信息,从而一直保持连接状态,

  • 而每一个ServerSocket线程它只能同时被一个输入流给占用,只可以一次与一个输出流保持正常连接状态

  • 所以在上一个输入流没有关闭且保持连接的情况下。下一个Socket对象获取的是下一个客户端的输出流对象

? 另一端接入的是上一个客户端的输入流,连接无法正常建立,所以即使你的数据是正常传输出去的

? 但是对应客户端的输入流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:这一次文章是自己第一次做出这样的尝试进行问题分析,有不对的地方希望各位大佬即使指出

  网络协议 最新文章
使用Easyswoole 搭建简单的Websoket服务
常见的数据通信方式有哪些?
Openssl 1024bit RSA算法---公私钥获取和处
HTTPS协议的密钥交换流程
《小白WEB安全入门》03. 漏洞篇
HttpRunner4.x 安装与使用
2021-07-04
手写RPC学习笔记
K8S高可用版本部署
mySQL计算IP地址范围
上一篇文章      下一篇文章      查看所有文章
加:2022-01-16 13:28:51  更:2022-01-16 13:29:26 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/8 5:53:23-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码