2024年4月16日发(作者:)

当服务器close一个连接时,若client端接着发数据。根据TCP协议的规定,会收到一

个RST响应,client再往这个服务器发送数据时,系统会发出一个SIGPIPE信号给进程,告

诉进程这个连接已经断开了,不要再写了。根据信号的默认处理规则SIGPIPE信号的默认

执行动作是 terminate(终止、退出), 所以client会退出。

若不想客户端退出可以把 SIGPIPE设为SIG_IGN

如: signal(SIGPIPE,SIG_IGN);

这时SIGPIPE交给了系统处理。

服务器采用了fork的话,要收集垃圾进程,防止僵死进程的产生,可以这样处理:

signal(SIGCHLD,SIG_IGN); 交给系统init去回收。

这里子进程就不会产生僵死进程了。

signal(SIGHUP, SIG_IGN);

signal信号函数,第一个参数表示需要处理的信号值(SIGHUP),第二个参数为处理函数

或者是一个表示,这里,SIG_IGN表示忽略SIGHUP那个注册的信号。

SIGHUP和控制台操作有关,当控制台被关闭时系统会向拥有控制台sessionID的所有进程

发送HUP信号,默认HUP信号的action是 exit,如果远程登陆启动某个服务进程并在程序运行

时关闭连接的话会导致服务进程退出,所以一般服务进程都会用nohup工具启动或写成一个

daemon。

unix中进程组织结构为 session 包含一个前台进程组及一个或多个后台进程组,一个进程

组包含多个进程。

一个session可能会有一个session首进程,而一个session首进程可能会有一个控制终端。

一个进程组可能会有一个进程组首进程。进程组首进程的进程ID与该进程组ID相等。

这儿是可能会有,在一定情况之下是没有的。

与终端交互的进程是前台进程,否则便是后台进程

SIGHUP会在以下3种情况下被发送给相应的进程:

1、终端关闭时,该信号被发送到session首进程以及作为job提交的进程(即用&符号提交

的进程)

2、session首进程退出时,该信号被发送到该session中的前台进程组中的每一个进程

3、若父进程退出导致进程组成为孤儿进程组,且该进程组中有进程处于停止状态(收到

SIGSTOP或SIGTSTP信号),该信号会被发送到该进程组中的每一个进程。

系统对SIGHUP信号的默认处理是终止收到该信号的进程。所以若程序中没有捕捉该信号,

当收到该信号时,进程就会退出。

表头文件 #include

功能:

设置某一信号的对应动作

函数原型 :

void (*signal(int signum,void(* handler)(int)))(int);

或者:typedef void(*sig_t) ( int ); sig_t signal(int signum,sig_t handler);

参数说明:

第一个参数signum指明了所要处理的信号类型,它可以取除了SIGKILL和SIGSTOP外的

任何一种信号。

第二个参数handler描述了与信号关联的动作,它可以取以下三种值:

(1)一个返回值为正数的函数地址 此函数必须在signal()被调用前申明,handler中为

这个函数的名字。当接收到一个类型为sig的信号时,就执行handler 所指定的函数。这个

函数应有如下形式的定义: intfunc(int sig); sig是传递给它的唯一参数。执行了

signal()调用后,进程只要接收到类型为sig的信号,不管其正在执行程序的哪一部分,就立

即执行func()函数。当func()函数执行结束后,控制权返回进程被中断的那一点继续执行。

(2)SIGIGN 这个符号表示忽略该信号,执行了相应的signal()调用后,进程会忽略类

型为sig的信号。

(3)SIGDFL 这个符号表示恢复系统对信号的默认处理。

函数说明 : signal()会依参数signum 指定的信号编号来设置该信号的处理函数。当指定

的信号到达时就会跳转到参数handler指定的函数执行。当一个信号的信号处理函数执行时,

如果进程又接收到了该信号,该信号会自动被储存而不会中断信号处理函数的执行,直到信

号处理函数执行完毕再重新调用相应的处理函数。但是如果在信号处理函数执行时进程收到

了其它类型的信号,该函数的执行就会被中断。

返回值: 返回先前的信号处理函数指针,如果有错误则返回SIG_ERR(-1)。

附加说明 :在信号发生跳转到自定的handler处理函数执行后,系统会自动将此处理函数换

回原来系统预设的处理方式,如果要改变此操作请改用sigaction()。

下面的情况可以产生Signal:

1. 按下CTRL+C产生SIGINT

2. 硬件中断,如除0,非法内存访问(SIGSEV)等等

3. Kill函数可以对进程发送Signal

4. Kill命令。实际上是对Kill函数的一个包装

5. 软件中断。如当Alarm Clock超时(SIGURG),当Reader中止之后又向管道写数据