Linux下忽略信号SIGPIPE的方法

转载自:https://blog.csdn.net/woxiaozhi/article/details/40624033#commentBox
最近为测试自己写的服务器,临时写了一个客户端,总是发现客户端收到SIGPIPE的信号,然后进程退出。

为了客户端进程收到SIGPIPE不退出,我打算忽略该信号,下面是我用过的方法:

(1)间接忽略

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)直接忽略

signal(SIGPIPE,SIG_IGN);  

(3)重载signaction

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被忽略,客户端没有因为该信号退出。

将这次经历记录下来,与看到的人共享。

猜你喜欢

转载自blog.csdn.net/topsluo/article/details/80780937