牛骨文教育服务平台(让学习变的简单)
博文笔记

C++SIGUSR1和SIGUSR2的学习

创建时间:2016-02-06 投稿人: 浏览次数:1462

代码转载自:http://www.sharejs.com/codes/cpp/6846
http://www.cnblogs.com/chenyadong/archive/2011/10/11/2207985.html
本文仅作自己学习记录,不作它用!
(一)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 SIGUSR1n");
    case SIGUSR2: //处理信号 SIGUSR2
        printf("Child : catch SIGUSR2n");
    default:      //本例不支持
        printf("Should not be heren");
        break;
    }
}

int main(void)
{
    pid_t ppid, cpid;
    //为两个信号设置信号处理函数
    if(signal(SIGUSR1, handler) == SIG_ERR) { //设置出错
        perror("Can"t set handler for SIGUSR1");
        exit(1);
    }

    if(signal(SIGUSR2, handler) == SIG_ERR) { //设置出错
        perror("Can"t set handler for SIGUSR2");
        exit(1);
    }

    ppid = getpid();//得到父进程ID

    if((cpid = fork()) < 0) {
        perror("fail to fork");
        exit(1);
    } else if(cpid == 0) { //子进程
        if(kill(ppid, SIGUSR1) == -1) {
            perror("fail to send signal");
            exit(1);
        }

        while(1);//死循环,等待父进程的信号
    } else {
        sleep(1);//休眠,保证子进程先运行,并且发送SIGUSR1信号

        if(kill(ppid, SIGUSR2) == -1) {
            perror("fail to send signal");
            exit(1);
        }

        printf("kill childn");//输出提示

        if(kill(ppid, SIGKILL) == -1) { //发送SIGKILL信号,杀死子进程
            perror("fail to send signal");
            exit(1);
        }

        if(wait(NULL) ==-1) { //回收子进程状态,避免僵尸进程
            perror("fail to wait");
            exit(1);
        }
    }

    return;
}
//该代码片段来自于: http://www.sharejs.com/codes/cpp/6846

(二)捕捉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"t catch SIGUSR1
");
        if(signal(SIGUSR2, sig_usr) == SIG_ERR)
                printf("can"t catch SIGUSR2
");
        for(; ;)
                pause();
}

static void sig_usr(int signo)
{
        if(signo == SIGUSR1)
                printf("received SIGUSR1
");
        else if(signo == SIGUSR2)
                printf("received SIGUSR2
");
        else
                printf("received signal %d
", 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

说明:
void sig_usr(int)为signal handler函数,用于注册信号,其实现也只是将接收到的信号打印出来而已,对SIGUSR1和SIGUSR2两种信号特别一些,直接打印出其信号名。

在main函数中,先是注册了两个信号:SIGUSR1和SIGUSR2,然后是一个死循环,使用pause不断的接收信号,并没有退出循环的相应机制,只能使用kill命令结束进程。

于是到现在,这个小程序便比较容易理解了,程序先使用signal函数将SIGUSR1和SIGUSR2两个信号注册到sig_usr信号处理函数上去,而信号处理函数sig_usr对信号的处理策略为:如果是SIGUSR1或SIGUSR2两种信号,打印出其信号类型,否则打印其信号值。

无意间,我们已经做了一次进程间通信:执行kill命令的进程将SIGUSR1或SIGUSR2信号传送给了进程id为2581的进程,而进程2581收到信号过后,按照约定(注册的信号处理函数)进行了相应的处理(打印进程信息)。

声明:该文观点仅代表作者本人,牛骨文系教育信息发布平台,牛骨文仅提供信息存储空间服务。