SO_LINGER 选项是在 BSD sockets API 中的一个套接字选项,用于控制 TCP 套接字在关闭时的行为。当应用程序调用 close()
或 shutdown()
函数关闭一个套接字时,如果套接字缓冲区中仍有待发送的数据(即所谓的“残余数据”),操作系统通常会尽可能地将这些数据发送出去,这一过程被称为正常关闭(Graceful Shutdown)。然而,这可能导致程序在关闭套接字后还需要等待一定时间才能真正退出,尤其是当对方主机不配合关闭握手时。
SO_LINGER 选项允许应用程序定制这种行为。通过设置 setsockopt()
函数的 SO_LINGER 选项,可以传入一个 linger
结构体,该结构体包含两个字段:
-
l_onoff
:一个布尔标志,决定是否启用 SO_LINGER 选项。如果设置为0,则禁用 SO_LINGER 选项,关闭套接字时会立即返回,而不考虑是否有残余数据;如果设置为非0,则启用 SO_LINGER。 -
l_linger
:如果l_onoff
为非0,则l_linger
字段指示了在关闭套接字之前应该等待多长时间(以秒为单位),等待期间会尝试将所有待发送的数据发送完毕。如果这段时间过后仍然有数据未发送完,则系统将会发送一个 RST(Reset)报文强制关闭连接,而不是等待 FIN-WAIT-2 状态下的超时重传。
在某些情况下,如需快速关闭套接字并忽略残余数据,可以通过设置 SO_LINGER 选项并设置较短的 linger 时间来实现。而在需要确保所有数据可靠发送的情况下,则应设置较大的 linger 时间或保留默认值,让操作系统负责正常的关闭序列。