socket中close发生的事情,RST,pipe信号错误

1、server端close之后,client端write,导致server端发送RST(服务器关闭套接字):对方已经关闭或者异常终止,但是client端,不知道,这个成为半打开

当server端close套接字的时候,假设此时server端的接受缓冲区没有数据了。则close发送的是FIN分节,client端如果收到FIN之后,调用read函数,是返回0的,因为FIN的接收,表明client端以后再无数据可以接收,因为对放发来FIN,就表明对方不在发送数据了。

然后client发送应答ack报文,表明收到FIN报文,server收到ack报文之后,就进去了FIN_WAIT_2阶段,根据tcp协议,向一个 FIN_WAIT2 状态的服务 TCP(已 ACK 响应 FIN 分节)写入数据不成问题,所以此时我们可以在client端调用write函数,写入client端的发送缓冲区,由tcp连接,发送到server的接收缓冲区。但是此时,由于server端已经关闭了socket,所以此时的接收缓冲区的内容都被抛弃,同时server端返回RST

收到RST的client端,如果调用read函数,读取,是返回RST错误的。。

如果我们的client对这个RST错误,不采取处理,继续调用write,会发生epipe错误。

这种情况一般发生在客户进程不理会(或未及时处理)Socket 错误,继续向服务 TCP 写入更多数据时,内核将向客户进程发送 SIGPIPE 信号,该信号默认会使进程终止(此时该前台进程未进行 core dump)。结合上边的 ECONNRESET 错误可知,向一个 FIN_WAIT2 状态的服务 TCP(已 ACK 响应 FIN 分节)写入数据不成问题,但是写一个已接收了 RST 的 Socket 则是一个错误。

接收到RST包的一端采取的操作:

异常关闭一个连接对应用程序来说有两个优点:

(1)丢弃任何待发的已经无意义的 数据,并立即发送RST报文段;

(2)RST的接收方利用关闭方式来 区分另一端执行的是异常关闭还是正常关闭(通过read操作的返回值,正常的返回读取字节或者堵塞,RST的返回错误)

值得注意的是RST报文段不会导致另一端产生任何响应(不用发送ack响应分节),另一端根本不进行确认。收到RST的一方将终止该连接。程序行为如下:

阻塞模型下,内核无法主动通知应用层出错,只有应用层主动调用read()或者write()这样的IO系统调用时,内核才会利用出错来通知应用层对端RST。

非阻塞模型下,select或者epoll会返回sockfd可读,应用层对其进行读取时,read()会报错RST。

2、server端close套接字,此时server端的接收缓冲区还有数据,没有被读取,则此时server端发送RST给client端,接收缓冲区的数据丢失,服务器server提前关闭socket

3、到不存在的端口连接请求

client端发起连接请求,发送了SYN报文,由于server端的套接字没有listen端口,则向cllient发送RST

4、同时主动打开connect()

5、同时主动关闭close()

时间: 2024-08-07 21:18:20

socket中close发生的事情,RST,pipe信号错误的相关文章

Java socket中关闭IO流后,发生什么事?(以关闭输出流为例)

声明:该博文以socket中,关闭输出流为例进行说明. 为了方便讲解,我们把DataOutputstream dout = new DataOutputStream(new BufferedOutputStream(mySocket.getOutputStream()));中的dout做为Socket输出流的代言.同样的,din是输入流的代言. 可以造成dout被关闭的操作有: 1.调用dout.close();或din.close();因为使用这种流关闭,会造成socket被关闭,所以输入输出

socket中的SIGPIPE信号问题

最近在做产品的老化测试,大体的方法是让产品长时间跑,数据由网络发出,PC端插入网线,则收到产品的数据,看是否正确,拔掉网线,产品端产生的数据则抛空,产品继续长时间运行. 想到的实现方法是产品做服务端,PC做客户端.在产品端重新实现open write close,在open中起子线程监听连接.write中如果有有效连接则写到socket中,无效连接则直接返回. 这样存在的一个问题是对于客户端的暴力关闭,也就是直接拔掉网线,服务器端如何检测. 也就是socket连接的一端如何知道连接已经被动关闭,

Linux基础概念-----touch一个新文件背后发生的事情

一个普通用户在Shell下执行了一条命令,这背后发生了什么事情,是个人的一些总结,如果有不对的地方欢迎搭建指正,谢谢! touch /tmp/test.txt ####################################################################################### tom在Shell中执行这条命令,Shell将命令发给内核,并且判断命令是否存在,通过文件系统上的元数据判断tom是否有权限执行,内核从内存中加载出mkdir的指令

在浏览器输入网址,Enter之后发生的事情

在浏览器输入网址,Enter之后发生的事情: 1. 浏览器接收域名 2. 发送域名给DNS,中文名字是域名系统服务器,一般位于ISP(互联网服务提供商,比如我们熟知的联通.移动.电信等) 中.浏览器会首先发给离自己最近的DNS,DNS收到浏览器发来的域名,在自己的数据库和缓存中查找这个域名所对应的IP,如果没有他会询问其他的DNS,递归下去直到弄清楚IP.最后把结果返回给浏览器. 3. DNS返回域名所对应的IP地址 4. 浏览器向因特网中发出请求 5. 路由器依据IP地址,把包裹送达IP所对应

ZeroMQ接口函数之 :zmq_msg_recv - 从一个socket中接受一个消息帧

ZeroMQ 官方地址 :http://api.zeromq.org/4-2:zmq_msg_recv zmq_msg_recv(3) ØMQ Manual - ØMQ/3.2.5 Name zmq_msg_recv - 从一个socket中接受一个消息帧 Synopsis int zmq_msg_recv (zmq_msg_t *msg, void *socket, int flags); Description zmq_msg_recv()函数和zmq_recvmsg(3)函数是完全相同的,

创建一个对象都在内存中做了什么事情

创建一个对象都在内存中做了什么事情? 1:先将硬盘上指定位置的Person.class文件加载进内存. 2:执行main方法时,在栈内存中开辟了main方法的空间(压栈-进栈),然后在main方法的栈区分配了一个变量p. 3:在堆内存中开辟一个实体空间,分配了一个内存首地址值.new 4:在该实体空间中进行属性的空间分配,并进行了默认初始化. 5:对空间中的属性进行显示初始化. 6:进行实体的构造代码块初始化. 7:调用该实体对应的构造函数,进行构造函数初始化.() 8:将首地址赋值给p ,p变

Student s = new Student();在内存中做了哪些事情?

[Student s = new Student();在内存中做了哪些事情?] 1加载Student.class文件进内存 2在栈内存为s开辟空间 3在堆内存为学生对象开辟空间 4对学生对象的成员变量进行默认初始化 5对学生对象的成员变量进行显示初始化 6 通过构造代码块对学生对象进行初始化(若没有就不执行) 7通过构造方法对学生对象进行初始化 (通过构造方法对学生对象的成员变量赋值)    对象初始化完毕. 8学生对象初始化完毕,把对象地址赋值给s变量,让变量s指向学生对象.

linux socket中的SO_REUSEADDR

Welcome to the wonderful world of portability... or rather the lack of it. Before we start analyzing these two options in detail and take a deeper look how different operating systems handle them, it should be noted that the BSD socket implementation

测试Socket中的一个中断的方法

/* * 本程序用于测试Socket中的一个中断的方法 * 对于进行的线程可以对其进行停止的中断 * 是通过SocketChannel channel = SocketChannel.open(new InetSocketAddress(host, port)); * */ package interruptible; import java.awt.BorderLayout; import java.awt.EventQueue; import java.awt.event.ActionEve