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

简介:
原代码片段如下,程序在第08行报错,但是try,,,catch无法抓到错误,经过debug分析是由于收到broken pipe的信号。看来程序此时被终止了;那么我期望不被终止,该怎么做了。

01 int sendLen = 0;
02 int totalLen = 0;
03 int packSize = pack.size();
04 while(packSize != totalLen)
05 {
06 try
07 {
08 sendLen = write(fd, const_cast<char*>(pack.c_str())+totalLen, packSize-totalLen);
09 totalLen += sendLen;
10 if(sendLen <= 0)
11 {
12 totalLen == 0;
13 fprintf(stderr,"write fd err . fd == %d - %m/n",fd);
14 return false;
15 }
16 }
17 catch (std::exception& e)
18 {
19 printf("errrno is:%d",errno);
20 std::cout << "Exception: " << e.what();
21 }
22 }
23 return true;
在徐明刚的指导下,找到如下文章
网摘
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.程序便能知道对端已经关闭.
为此代码变更如下;插入10行
01 bool CWWSimulator::SendPack(string& pack, uint32_t fd)
02 { 
03 int sendLen = 0;
04 int totalLen = 0;
05 int packSize = pack.size();
06 while(packSize != totalLen)
07 {
08 try
09 {
10 signal(SIGPIPE, SIG_IGN);
11 sendLen = write(fd, const_cast<char*>(pack.c_str())+totalLen, packSize-totalLen);
12 totalLen += sendLen;
13 if(sendLen <= 0)
14 {
15 totalLen == 0;
16 fprintf(stderr,"write fd err . fd == %d - %m/n",fd);
17 return false;
18 }
19 }
20 catch (std::exception& e)
21 {
22 printf("errrno is:%d",errno);
23 std::cout << "Exception: " << e.what();
24 }
25 } 
26 return true;
27 }


本文转自elbertchen 51CTO博客,原文链接:http://blog.51cto.com/linkyou/751877,如需转载请自行联系原作者
相关文章
|
2月前
|
网络协议 安全 Linux
Linux C/C++之IO多路复用(select)
这篇文章主要介绍了TCP的三次握手和四次挥手过程,TCP与UDP的区别,以及如何使用select函数实现IO多路复用,包括服务器监听多个客户端连接和简单聊天室场景的应用示例。
98 0
|
2月前
|
存储 Linux C语言
Linux C/C++之IO多路复用(aio)
这篇文章介绍了Linux中IO多路复用技术epoll和异步IO技术aio的区别、执行过程、编程模型以及具体的编程实现方式。
106 1
Linux C/C++之IO多路复用(aio)
|
2月前
|
Ubuntu Linux 编译器
Linux/Ubuntu下使用VS Code配置C/C++项目环境调用OpenCV
通过以上步骤,您已经成功在Ubuntu系统下的VS Code中配置了C/C++项目环境,并能够调用OpenCV库进行开发。请确保每一步都按照您的系统实际情况进行适当调整。
503 3
|
2月前
|
资源调度 Linux 调度
Linux C/C++之线程基础
这篇文章详细介绍了Linux下C/C++线程的基本概念、创建和管理线程的方法,以及线程同步的各种机制,并通过实例代码展示了线程同步技术的应用。
33 0
Linux C/C++之线程基础
|
2月前
|
Linux C++
Linux C/C++之IO多路复用(poll,epoll)
这篇文章详细介绍了Linux下C/C++编程中IO多路复用的两种机制:poll和epoll,包括它们的比较、编程模型、函数原型以及如何使用这些机制实现服务器端和客户端之间的多个连接。
36 0
Linux C/C++之IO多路复用(poll,epoll)
|
2月前
|
网络协议 Linux 网络性能优化
Linux C/C++之TCP / UDP通信
这篇文章详细介绍了Linux下C/C++语言实现TCP和UDP通信的方法,包括网络基础、通信模型、编程示例以及TCP和UDP的优缺点比较。
47 0
Linux C/C++之TCP / UDP通信
|
2月前
|
消息中间件 Linux API
Linux c/c++之IPC进程间通信
这篇文章详细介绍了Linux下C/C++进程间通信(IPC)的三种主要技术:共享内存、消息队列和信号量,包括它们的编程模型、API函数原型、优势与缺点,并通过示例代码展示了它们的创建、使用和管理方法。
41 0
Linux c/c++之IPC进程间通信
|
2月前
|
Linux C++
Linux c/c++进程间通信(1)
这篇文章介绍了Linux下C/C++进程间通信的几种方式,包括普通文件、文件映射虚拟内存、管道通信(FIFO),并提供了示例代码和标准输入输出设备的应用。
38 0
Linux c/c++进程间通信(1)
|
2月前
|
Linux C++
Linux c/c++之进程的创建
这篇文章介绍了在Linux环境下使用C/C++创建进程的三种方式:system函数、fork函数以及exec族函数,并展示了它们的代码示例和运行结果。
47 0
Linux c/c++之进程的创建
|
2月前
|
Linux C++
Linux c/c++进程之僵尸进程和守护进程
这篇文章介绍了Linux系统中僵尸进程和守护进程的概念、产生原因、解决方法以及如何创建守护进程。
30 0
下一篇
DataWorks