Linux SIGPIPE信号

tcp 通信双方中,为了描述方便,以下将通信双方用 a 和 b 代替。

根据TCP协议规定,如果A关闭连接后B继续发送数据,B会收到A的RST响应。若B继续发送数据,系统会发出SIGPIPE信号告知连接已断开,停止发送。

系统对 SIGPIPE 信号的默认处理行为是让 B 进程退出。

操作系统对 SIGPIPE 信号的这种默认处理行为非常不友好,让我们来分析一下。

Linux SIGPIPE信号

TCP通信是全双工信道,相当于两条单工信道,连接两端各负责一条。

当对端“关闭”时, 虽然本意是关闭整个两条信道,但本端只是收到 FIN 包。

根据TCP协议的规定,当一端关闭其负责的单向通道时,仍可接收数据但不再发送数据。

也就是说,因为 TCP 协议的限制,通信一方无法获知对端的 socket 是调用了 close 还是 shutdown。

int shutdown(int socket, int how);

登录后复制

shutdown 函数的参数 how 可以设置为关闭 SHUT_RD、SHUT_WR 或 SHUT_RDWR 用于表示关闭收、发单个通道或者同时关闭收发通道。

对一个已经收到 FIN 包的 socket 调用 read/recv 方法, 如果接收缓冲已空,则返回 0,这就是常说的表示连接关闭。但第一次对其调用 write/send 方法时,如果发送缓冲没问题,会返回正确写入(即 write/send 函数返回值大于 0),但发送的报文会导致对端回应 RST 报文。因为上一次程序调用 write/send 是正常的,再次尝试调用 write/send 函数时因产生 SIGPIPE 信号导致进程退出。

这种默认行为对于我们开发程序,尤其是对于后端服务,需要同时对许多客户端服务,不能因为与某一个客户端的连接出问题了而导致整个进程退出不能继续为其他客户端服务。

为了避免这种现象出现, 可以捕获 SIGPIPE 信号并对其进行处理或者忽略该信号, 忽略该信号代码如下:

signal(SIGPIPE, SIG_IGN);

登录后复制

这样设置后,第二次调用 write/send 方法时,会返回 -1,同时 errno 错误码被置为 SIGPIPE,程序便能知道对端已经关闭。

以上就是Linux SIGPIPE信号的详细内容,更多请关注【创想鸟】其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至253000106@qq.com举报,一经查实,本站将立刻删除。

发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/2291403.html

(0)
上一篇 2025年2月28日 00:59:39
下一篇 2025年2月26日 23:59:12

AD推荐 黄金广告位招租... 更多推荐

相关推荐

发表回复

登录后才能评论