调用closesocket()背后发生的事

应用层代码想要关闭一个TCP连接可以直接调用closesocket(),但是这个函数的行为并不“单纯”,只有了解closesocket()背后的故事,才能对关闭TCP连接有正确的认识。

首先明确一点:任何情况下,调用closesocket之后,socket句柄立即失效(一种情况除外,见下文),再对该socket进行操作的API都会失败。但是socket句柄对应的系统资源不一定立即释放,由内核控制。

假定A端调用closesocket(),进行主动关闭。以下是几种情况:

1、默认的行为:尽可能的尝试完成4次握手(互相的FIN与ACK),此时,A端发送缓冲区的数据会发送到B端,并等待B端的确认,B端发送FIN后,A端进入TIME_WAIT状态。

2、l_onoff值为0:相当于默认行为。

3、l_onoff值非0,l_linger为0:A端调用closesocket()将发送RST复位连接,所有发送、接收缓冲区的数据都被丢弃,socket资源被释放,A端不进入TIME_WAIT状态。B端的recv会得到WSAECONNRESET错误。

4、l_onoff值非0,l_linger非0,socket是阻塞的:则closesocket()不会立即返回,而是等待l_linger长的时间,在l_linger时间内如果发送缓冲区里的数据发送并被B端确认,则closesocket()返回,之后任然是“优雅”的关闭,A端进入TIME_WAIT状态;如果发送缓冲区的数据没有发送完毕或者没有收到B端确认,则closesocket返回,同时内核放弃没有发送的数据或是不再等待B端的确认,直接发送RST复位连接,A端不进入TIME_WAIT状态。

5、l_onoff值非0,l_linger非0,socket是非阻塞的:调用closesocket()立即返回,如果从调用到返回的时间内,不能完成发送缓冲区数据的发送和对端的确认,则closesocket返回WSAEWOULDBLOCK;要注意的是,返回WSAEWOULDBLOCK不意味着socket句柄失效,仅意味着这次尝试closesocket不成功,应用层需要稍后继续调用closesocket()来尝试关闭连接(是不是觉得和非阻塞下的connect()、send()、recv()很像?:),直到closesocket()不返回WSAEWOULDBLOCK。

 

由以上各种情况也可以看出:应用层简单的closesocket()不能保证对端的应用层正确的收到数据(对端的接收缓冲区收到不代表应用层收到)。

  • 1
    点赞
  • 17
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值