linux下tcp服务端主动发送数据,Linux 下TCP连接关闭情况分析

一、TCP连接关闭的几种方式:

1、“正常”关闭:调用close()关闭socket、没close但进程正常结束(当然这是不应该的做法)、进程core掉、在shell命令 行中kill掉进程,都可抽象成“正常”关闭。因为即使core掉,内核也会马上帮应用程序回收(close)socket文件描述符。

“正常”关闭,默认情况下(非默认即设置Linger下面会介绍),关闭端即客户端TCP层会发FIN包,对端即服务器TCP层收到后,回ACK,客户端 进入FIN_WAIT2状态。此时,TCP终止连接的4个分组中服务器应该发的第3个分组FIN包,其TCP层是不会主动发的,只有服务器端socket “正常”关闭,才会发出这个FIN包。至此,客户端进入TIME_WAIT状态。

2、“非”正常关闭:客户端崩溃了,此时肯定发不出FIN包了(当然啦,内核都没机会帮应用程序回收资源了)。这种情况,服务器端有如下两种情况:

A、服务器send数据,因为客户端已经崩溃,服务器收不到ACK自然会不停的重传。源自

Berkeley的重传机制,重传8次,相对第一次传的15分钟后仍没收到ACK,则返回

ETIMEDOUT或EHOSTUNREAC错误。如果服务器不理会这个错误,再次调用send,则

立马返回Broken Pipe错误。

注:15分钟超时可以在 /proc/sys/net/ipv4/tcp_retries2 中修改

B、 服务器不发任何数据了,那只有靠应用层心跳检测机制或Keepalive,来发觉TCP断连了。

二、SO_LINGER套接口选项

A、l_onoff设置为0,这也是默认情况,函数close()是立即返回的,然后TCP连接双方是通过

FIN、ACK4分组来终止TCP连接的。当然,发送缓冲区还有数据的话,系统将试着将这些数据

发送到对方。

B、l_onoff非0,l_linger设置0,函数close()立即返回,并发送RST终止连接,发送缓冲区的数据丢弃。

C、l_onoff非0,l_linger非0,函数close()不立即返回,而是在(a)发送缓冲区数据发送完并得到确认

(b)l_linger延迟时间到,l_linger时间单位为微妙。两者之一成立时返回。如果在发送缓冲区数据发送

完并被确认前延迟时间到的话,close返回EWOULDBLOCK(或EAGAIN)错误。

三、客户端TCP连接“正常”关闭,服务器的几种情况:

情形

客户端l_onoff设置为0,

“正常”关闭

客户端l_onoff非0,l_linger设置0,“正常”关闭

服务器阻塞模式send,正阻塞在send函数未返回

客户端TCP发送FIN,服务器send函数返回成功(返回字节数是实际拷贝到发送缓冲区的字节数)。客户端发送RST。如果服务器再次调用send,将返回errno[32]:Broken pipe

客户端TCP发送RST,服务器函数返回成功(返回字节数是实际拷贝到发送缓冲区的字节 数)。若服务器再次调用send,则返回-1,errno[104]:Connection reset by peer。若再次调用send,则返回-1,errno[32]:Broken pipe

服务器空闲

客户端TCP发送FIN,若服务器没理会而调用send,客户端发送RST,send返回-1,errno[32]:Broken pipe

客户端TCP发送RST,若服务器没理会而调用send,send返回-1,errno[104]:Connection reset by peer。若再次调用send,则返回-1,errno[32]:Broken pipe

总之,1、收到对端RST后,仍然调入send(),则返回Connection reset by peer,再次调用send(),则返回Broken pipe

2、收到对端FIN后,仍然调研哪个send(),直接返回Broken pipe

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值