Linux的SIGUSR1和SIGUSR2信号

Posted 代码如诗

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Linux的SIGUSR1和SIGUSR2信号相关的知识,希望对你有一定的参考价值。

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);
}
运行结果:
[[email protected] apue]$ ./a.out &
[1] 2581
[[email protected] apue]$ kill -USR1 2581
received SIGUSR1
[[email protected] apue]$ kill -USR2 2581
received SIGUSR2
[[email protected] apue]$ kill 2581
[1]+ Terminated              ./a.out

  

以上是关于Linux的SIGUSR1和SIGUSR2信号的主要内容,如果未能解决你的问题,请参考以下文章

转-Unix系统进程对SIGTERMSIGUSR1和SIGUSR2信号处理

C等待信号循环

Linux 信号详解

linux 的信号种类

python 信号通信

7. 信号的处理问题