SO_LINGER作用

SO_LINGER这个选项在我以前带队改造haproxy的时候引出过一个reset(RST)客户端连接的bug。

SO_LINGER作用
设置函数close()关闭TCP连接时的行为。缺省close()的行为是,如果有数据残留在socket发送缓冲区中则系统将继续发送这些数据给对方,等待被确认,然后返回。

利用此选项,可以将此缺省行为设置为以下两种
  a.立即关闭该连接,通过发送RST分组(而不是用正常的FIN|ACK|FIN|ACK四个分组)来关闭该连接。至于发送缓冲区中如果有未发送完的数据,则丢弃。主动关闭一方的TCP状态则跳过TIMEWAIT,直接进入CLOSED。网上很多人想利用这一点来解决服务器上出现大量的TIMEWAIT状态的socket的问题,但是,这并不是一个好主意,这种关闭方式的用途并不在这儿,实际用途在于服务器在应用层的需求。
  b.
将连接的关闭设置一个超时。如果socket发送缓冲区中仍残留数据,进程进入睡眠,内核进入定时状态去尽量去发送这些数据。
    在超时之前,如果所有数据都发送完且被对方确认,内核用正常的FIN|ACK|FIN|ACK四个分组来关闭该连接,close()成功返回。
    如果超时之时,数据仍然未能成功发送及被确认,用上述a方式来关闭此连接。close()返回EWOULDBLOCK。


SO_LINGER选项使用如下结构:

struct linger {

     int l_onoff;

     int l_linger;

};


l_onoff为0,则该选项关闭,l_linger的值被忽略,close()用上述缺省方式关闭连接。

l_onoff非0,l_linger为0,close()用上述a方式关闭连接。

l_onoff非0,l_linger非0,close()用上述b方式关闭连接。

值得一说的是,不管你用什么方式使用SO_LINGER,都需要大量耐心的测试来确保你的服务器程序确实是按照你的意愿在跑,因为这个选项对服务器处理小量请求的行为影响非常细微,简单的功能测试很难验证它的行为,上线后量大的情况下可能会出问题,让你的服务器马上下线,大并发量的模拟实际场景压测才能保证其正确性:)


SO_LINGER还有一个特点是可以用来控制TCP异常关闭需求。

终止一个连接的正常方式是发送FIN。在发送缓冲区中所有排队数据都已发送之后才发送FIN,正常情况下没有任何数据丢失。

但我们有时也有可能发送一个RST报文段而不是FIN来中途关闭一个连接。这称为异常关闭

进程关闭socket的默认方式是正常关闭,如果需要异常关闭,利用SO_LINGER选项来控制。


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

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

(2)RST的接收方利用关闭方式来区分另一端执行的是异常关闭还是正常关闭。


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

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

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


  • 15
    点赞
  • 58
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
`setsockopt` 函数用于设置套接字选项。`SO_LINGER` 是一个用于设置套接字关闭时的延迟关闭选项。当 `SO_LINGER` 被设置为非零值时,关闭套接字时将会等待一段时间,直到发送或接收缓冲区中的数据被发送或丢弃。这可以确保数据的可靠传输。当 `SO_LINGER` 被设置为零时,关闭套接字时将立即关闭,不管缓冲区中是否还有数据。 在 C 语言中,使用 `setsockopt` 函数来设置 `SO_LINGER` 选项的代码示例如下: ```c #include <sys/types.h> #include <sys/socket.h> int setsockopt(int sockfd, int level, int optname, const void *optval, socklen_t optlen); ``` 其中,`sockfd` 是套接字描述符,`level` 是选项的级别(通常为 `SOL_SOCKET`),`optname` 是要设置的选项名称(在这里是 `SO_LINGER`),`optval` 是一个指向设置选项值的缓冲区的指针,`optlen` 是缓冲区的长度。 要设置 `SO_LINGER` 选项,可以按照以下步骤进行: ```c #include <sys/types.h> #include <sys/socket.h> int sockfd; struct linger so_linger; // 创建套接字 sockfd = socket(AF_INET, SOCK_STREAM, 0); // 设置 SO_LINGER 选项 so_linger.l_onoff = 1; // 非零值表示启用 SO_LINGER so_linger.l_linger = 5; // 延迟关闭时间为 5 秒 setsockopt(sockfd, SOL_SOCKET, SO_LINGER, &so_linger, sizeof(so_linger)); // 关闭套接字 close(sockfd); ``` 在上面的示例中,我们创建了一个 TCP 套接字 `sockfd`,然后设置了 `SO_LINGER` 选项的值为非零,延迟关闭时间为 5 秒。最后,我们关闭了套接字。 请注意,`setsockopt` 函数返回值为 0 表示成功,-1 表示失败。你可以根据返回值进行错误处理。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值