SIGUSR1 用戶自定義信號 默認處理:進程終止
SIGUSR2 用戶自定義信號 默認處理:進程終止
當一個進程調用fork時,因為子進程在開始時復制父進程的存儲映像,信號捕捉函數的地址在子進程中是有意義的,所以子進程繼承父進程的信號處理方式。
但是當子進程調用exec后,因為exec運行新的程序后會覆蓋從父進程繼承來的存儲映像,那么信號捕捉函數在新程序中已無意義,所以exec會將原先設置為要捕捉的信號都更改為默認動作。
C++父子進程使用SIGUSR1和SIGUSR2進行通信
#include <stdio.h>
#include <stdlib.h>
#include <signal.h>
#include <unistd.h>
#include <sys/wait.h>
void handler(int signo)
{
switch(signo) {
case SIGUSR1: //處理信號 SIGUSR1
printf("Parent : catch SIGUSR1\n");
break;
case SIGUSR2: //處理信號 SIGUSR2
printf("Child : catch SIGUSR2\n");
break;
default: //本例不支持
printf("Should not be here\n");
break;
}
}
int main(void)
{
pid_t ppid, cpid;
//為兩個信號設置信號處理函數
if(signal(SIGUSR1, handler) == SIG_ERR)
{ //設置出錯
perror("Can't set handler for SIGUSR1\n");
exit(1);
}
if(signal(SIGUSR2, handler) == SIG_ERR)
{ //設置出錯
perror("Can't set handler for SIGUSR2\n");
exit(1);
}
ppid = getpid();//得到父進程ID
if((cpid = fork()) < 0)
{
perror("fail to fork\n");
exit(1);
}
else if(cpid == 0)
{
// 子進程內向父進程發送信號SIGUSER1
if(kill(ppid, SIGUSR1) == -1)
{
perror("fail to send signal\n");
exit(1);
}
while(1);//死循環,等待父進程的信號
}
else
{
sleep(1);//休眠,保證子進程先運行,並且發送SIGUSR1信號
// 父進程向自己發送SIGUSER2信號
if(kill(cpid, SIGUSR2) == -1)
{
perror("fail to send signal\n");
exit(1);
}
// 必須sleep一下,否則子進程捕獲不到SIGUSER2信號
sleep(1);
printf("will kill child\n");//輸出提示
if(kill(cpid, SIGKILL) == -1)
{ //發送SIGKILL信號,殺死子進程
perror("fail to send signal\n");
exit(1);
}
if(wait(NULL) ==-1)
{ //回收子進程狀態,避免僵屍進程
perror("fail to wait\n");
exit(1);
}
printf("child has been killed.\n");
}
return;
}
捕捉SIGUSR1和SIGUSR2的簡單程序
#include <stdio.h> #include <signal.h> #include <unistd.h> static void sig_usr(int); int main(void) { if(signal(SIGUSR1, sig_usr) == SIG_ERR) printf("can not catch SIGUSR1\n"); if(signal(SIGUSR2, sig_usr) == SIG_ERR) printf("can not catch SIGUSR2\n"); for(;;) pause(); } static void sig_usr(int signo) { if(signo == SIGUSR1) printf("received SIGUSR1\n"); else if(signo == SIGUSR2) printf("received SIGUSR2\n"); else printf("received signal %d\n", signo); }
運行結果:
[chinsung@thinkpad apue]$ ./a.out &
[1] 2581
[chinsung@thinkpad apue]$ kill -USR1 2581
received SIGUSR1
[chinsung@thinkpad apue]$ kill -USR2 2581
received SIGUSR2
[chinsung@thinkpad apue]$ kill 2581
[1]+ Terminated ./a.out