Linux SIGPIPE信号

王林
发布: 2024-02-19 16:00:33
转载
536人浏览过

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

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

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

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

Linux SIGPIPE信号

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

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

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

微信 WeLM
微信 WeLM

WeLM不是一个直接的对话机器人,而是一个补全用户输入信息的生成模型。

微信 WeLM 33
查看详情 微信 WeLM

也就是说,因为 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信号的详细内容,更多请关注php中文网其它相关文章!

相关标签:
最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:每日运维网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号