Linux下忽略信号SIGPIPE的方法

最近为测试自己写的服务器,临时写了一个客户端,总是发现客户端收到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被忽略,客户端没有因为该信号退出。

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


你可能感兴趣的:(网络笔记,C++学习笔记)