linux 下写socket遭遇broken pipe(SIGPIPE C++)

本文介绍如何处理Linux环境下Socket编程中出现的Broken Pipe错误。通过调整信号处理方式,避免程序因SIGPIPE信号而意外终止。提供了修改前后的代码示例,并解释了signal函数的作用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

<p>原代码片段如下,程序在第08行报错,但是try,,,catch无法抓到错误,经过debug分析是由于收到broken pipe的信号。看来程序此时被终止了;那么我期望不被终止,该怎么做了。</p>
<p><br>
01 int sendLen = 0;<br>
02 int totalLen = 0;<br>
03 int packSize = pack.size();<br>
04 while(packSize != totalLen)<br>
05 {<br>
06 try<br>
07 {<br>
08 sendLen = write(fd, const_cast<char*>(pack.c_str())+totalLen, packSize-totalLen);<br>
09 totalLen += sendLen;<br>
10 if(sendLen <= 0)<br>
11 {<br>
12 totalLen == 0;<br>
13 fprintf(stderr,"write fd err . fd == %d - %m/n",fd);<br>
14 return false;<br>
15 }<br>
16 }<br>
17 catch (std::exception& e)<br>
18 {<br>
19 printf("errrno is:%d",errno);<br>
20 std::cout << "Exception: " << e.what();<br>
21 }<br>
22 }<br>
23 return true;</p>
<p>在徐明刚的指导下,找到如下文章</p>
<p>网摘</p>
<p> linux下写socket的程序的时候,如果尝试send到一个disconnected socket上,就会让底层抛出一个SIGPIPE信号。
client端通过 pipe 发送信息到server端后,就关闭client端, 这时server端,返回信息给 client 端时就产生Broken pipe 信号了。
对于产生信号,我们可以在产生信号前利用方法 signal(int signum, sighandler_t handler) 设置信号的处理。如果没有调用此方法,系统就会调用默认处理方法:中止程序,显示提示信息(就是我们经常遇到的问题)。我们可以调用系统的处理方法,也可以自定义处理方法。
对一个已经收到FIN包的socket调用read方法, 如果接收缓冲已空, 则返回0,
这就是常说的表示连接关闭. 但第一次对其调用write方法时, 如果发送缓冲没问题,
会返回正确写入(发送). 但发送的报文会导致对端发送RST报文,
因为对端的socket已经调用了close, 完全关闭, 既不发送, 也不接收数据. 所以,
第二次调用write方法(假设在收到RST之后), 会生成SIGPIPE信号, 导致进程退出.
为了避免进程退出, 可以捕获SIGPIPE信号, 或者忽略它,
给它设置SIG_IGN信号处理函数:
signal(SIGPIPE, SIG_IGN);
这样, 第二次调用write方法时, 会返回-1, 同时errno置为SIGPIPE.
程序便能知道对端已经关闭.</p>
<p></p>
<p>为此代码变更如下;插入10行</p>
<p>01 bool CWWSimulator::SendPack(string& pack, uint32_t fd)<br>
02 { <br>
03 int sendLen = 0;<br>
04 int totalLen = 0;<br>
05 int packSize = pack.size();<br>
06 while(packSize != totalLen)<br>
07 {<br>
08 try<br>
09 {<br>
10 signal(SIGPIPE, SIG_IGN);<br>
11 sendLen = write(fd, const_cast<char*>(pack.c_str())+totalLen, packSize-totalLen);<br>
12 totalLen += sendLen;<br>
13 if(sendLen <= 0)<br>
14 {<br>
15 totalLen == 0;<br>
16 fprintf(stderr,"write fd err . fd == %d - %m/n",fd);<br>
17 return false;<br>
18 }<br>
19 }<br>
20 catch (std::exception& e)<br>
21 {<br>
22 printf("errrno is:%d",errno);<br>
23 std::cout << "Exception: " << e.what();<br>
24 }<br>
25 } <br>
26 return true;<br>
27 }</p>
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值