线程中解决sigpipe信号问题
2018-02-07 17:08
281 查看
有时候线程会出现SIGPIPE错误:
(gdb) bt
#0 0x00110416 in __kernel_vsyscall ()
#1 0x0092f918 in send () from /lib/libpthread.so.0
这是因为在调用socket发送函数send()时,出现的异常错误,可能的原因是因为tcp连接可能不正常,导致了send()数据失败,系统会收到SIGPIPE信号,
系统默认的处理是退出进程。可以让系统屏蔽SIGPIPE信号,这样服务端进程就不会异常退出了,可以继续运行。
1.在main函数一开始执行下面的屏蔽SIGPIPE信号:
signal(SIGPIPE, SIG_IGN);
2. 但在线程中还是会出现SIGPIPE问题,添加下面的代码,屏蔽线程中的SIGPIPE信号:
sigset_t signal_mask;
sigemptyset (&signal_mask);
sigaddset (&signal_mask, SIGPIPE);
int rc = pthread_sigmask (SIG_BLOCK, &signal_mask, NULL);
if (rc != 0)
{
printf("block sigpipe error\n");
}
上面的代码应写在main函数的开头,创建线程之前。
问题解决。
转自:http://blog.csdn.net/doctorllm/article/details/43410027
最近为测试自己写的服务器,临时写了一个客户端,总是发现客户端收到SIGPIPE的信号,然后进程退出。
为了客户端进程收到SIGPIPE不退出,我打算忽略该信号,下面是我用过的方法:
(1)间接忽略
[cpp] view
plain copy
static void SignalHandler(int nSigno)
{
signal(nSigno, SignalHandler);
switch(nSigno)
{
case SIGPIPE:
printf("Process will not exit\n");
break;
default:
printf("%d signal unregister\n", nSigno);
break;
}
}
atic void InitSignalHandler()
{
signal(SIGPIPE , &SignalHandler);
}
int main()
{
InitSignalHandler();
........
return 0;
}
(2)直接忽略
[cpp] view
plain copy
signal(SIGPIPE,SIG_IGN);
(3)重载signaction
[cpp] view
plain copy
struct sigaction sa;
sa.sa_handler = SIG_IGN;
sigaction( SIGPIPE, &sa, 0 );
(1)(2)(3)都不能够忽略,客户端收到SIGPIPE之后依然进程退出。
于是乎,查SIGPIPE这个信号的特性:
如果在写到管道时读进程已终止,则产生此信号。当类型为SOCK_STREAM的套接字已不再连接时,进程写到该套接字也产生此信号。
——《UNIX环境高级编程》中10.2节
由于我的客户端是用send()进行发送数据的,通过man手册查看send()函数,看到有一条这样说:
MSG_NOSIGNAL
Requests not to send SIGPIPE on errors on stream oriented sockets when the other end breaks the connection. The EPIPE error is still returned.
于是将send()最后一个参数flags改为MSG_NOSIGNAL,再次启动客户端测试。SIGPIPE被忽略,客户端没有因为该信号退出。
转自: http://blog.csdn.net/woxiaozhi/article/details/40624033
(gdb) bt
#0 0x00110416 in __kernel_vsyscall ()
#1 0x0092f918 in send () from /lib/libpthread.so.0
这是因为在调用socket发送函数send()时,出现的异常错误,可能的原因是因为tcp连接可能不正常,导致了send()数据失败,系统会收到SIGPIPE信号,
系统默认的处理是退出进程。可以让系统屏蔽SIGPIPE信号,这样服务端进程就不会异常退出了,可以继续运行。
1.在main函数一开始执行下面的屏蔽SIGPIPE信号:
signal(SIGPIPE, SIG_IGN);
2. 但在线程中还是会出现SIGPIPE问题,添加下面的代码,屏蔽线程中的SIGPIPE信号:
sigset_t signal_mask;
sigemptyset (&signal_mask);
sigaddset (&signal_mask, SIGPIPE);
int rc = pthread_sigmask (SIG_BLOCK, &signal_mask, NULL);
if (rc != 0)
{
printf("block sigpipe error\n");
}
上面的代码应写在main函数的开头,创建线程之前。
问题解决。
转自:http://blog.csdn.net/doctorllm/article/details/43410027
最近为测试自己写的服务器,临时写了一个客户端,总是发现客户端收到SIGPIPE的信号,然后进程退出。
为了客户端进程收到SIGPIPE不退出,我打算忽略该信号,下面是我用过的方法:
(1)间接忽略
[cpp] view
plain copy
static void SignalHandler(int nSigno)
{
signal(nSigno, SignalHandler);
switch(nSigno)
{
case SIGPIPE:
printf("Process will not exit\n");
break;
default:
printf("%d signal unregister\n", nSigno);
break;
}
}
atic void InitSignalHandler()
{
signal(SIGPIPE , &SignalHandler);
}
int main()
{
InitSignalHandler();
........
return 0;
}
(2)直接忽略
[cpp] view
plain copy
signal(SIGPIPE,SIG_IGN);
(3)重载signaction
[cpp] view
plain copy
struct sigaction sa;
sa.sa_handler = SIG_IGN;
sigaction( SIGPIPE, &sa, 0 );
(1)(2)(3)都不能够忽略,客户端收到SIGPIPE之后依然进程退出。
于是乎,查SIGPIPE这个信号的特性:
如果在写到管道时读进程已终止,则产生此信号。当类型为SOCK_STREAM的套接字已不再连接时,进程写到该套接字也产生此信号。
——《UNIX环境高级编程》中10.2节
由于我的客户端是用send()进行发送数据的,通过man手册查看send()函数,看到有一条这样说:
MSG_NOSIGNAL
Requests not to send SIGPIPE on errors on stream oriented sockets when the other end breaks the connection. The EPIPE error is still returned.
于是将send()最后一个参数flags改为MSG_NOSIGNAL,再次启动客户端测试。SIGPIPE被忽略,客户端没有因为该信号退出。
转自: http://blog.csdn.net/woxiaozhi/article/details/40624033
相关文章推荐
- 线程中解决sigpipe信号问题
- SIGPIPE信号--解决Linux socket中接收端退出导致发送端send退出问题
- 解决iOS11图片下拉放大出现信号栏白条的bug问题
- 转:C#中跨线程访问控件问题解决方案
- 用sigsetjmp和siglongjmp 解决从信号处理程序返回后信号仍被屏蔽问题
- 玩转Web之servlet(五)---- 怎样解决servlet的线程安全问题
- 查遍了所有的大富翁和CSDN的贴子,也没能解决的一个线程问题,不要笑我为啥这样做,我只求解决
- C# Winform使用线程,委托定时更新界面UI控件,解决界面卡顿问题
- 程序被SIGPIPE信号终止的原因以及解决方法
- java 多线程学习之多生产者多消费者产生的线程安全问题分析与解决:Lock和Condition
- 【极客学院】-python学习笔记-3-单线程爬虫 (request安装遇到问题及解决,应用requests提取信息)
- 关于windebug查询进程中各个线程占用cpu时间,解决CPU占用很大问题
- 基础篇_线程 第5集 多线程的安全问题--解决之道同步函数
- Qt信号函数无法解析的问题解决方法
- C#中跨线程访问控件问题解决方案
- 多线程(线程间通信-多生产者多消费者问题-JDK1.5解决办法-范例),停止线程,线程中方法的区别,匿名内部类实现多线程,线程总结
- 泛泰A850L清基带缓存工具,可解决莫名其妙的信号问题,只适用于泛泰A850L
- 图像检索服务器编写问题记录——线程中信号的处理方式思考
- 深入学习理解java:高效的解决死锁问题的线程通讯方式:Semaphore 和 BlockingQueue
- invoke委托解决“线程间操作无效: 从不是创建控件的线程访问它”的问题