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

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


GitHub 加速计划 / li / linux-dash
10.39 K
1.2 K
下载
A beautiful web dashboard for Linux
最近提交(Master分支:2 个月前 )
186a802e added ecosystem file for PM2 4 年前
5def40a3 Add host customization support for the NodeJS version 4 年前
Logo

旨在为数千万中国开发者提供一个无缝且高效的云端环境,以支持学习、使用和贡献开源项目。

更多推荐