《UNIX环境高级编程》笔记--可靠信号

原创 2013年12月06日 15:44:17

在产生信号时,内核通常在进程表中设置某一种形式的标志。当对信号采取了这种动作时,我们说向进程递送了一个信号。

在信号产生和递送之间的时间间隔内,称信号是未决的。

进程可以选用信号递送阻塞。如果为进程产生了一个选择为阻塞的信号,而且对该进程的动作是默认动作或捕捉该信号,则

为该进程将次信号保持为未决状态,直到该进程(1)对此信号解除了阻塞,或者(2)将次信号的动作更改为忽略。内核在

递送一个原来被阻塞的信号给进程时(而不是在产生该信号时),才决定对它的处理方式。于是子进程在信号递送给他之前

仍可改变对该信号的动作。

信号在内核中的表示可以看作是这样的:


每个信号都有两个标志位分别表示阻塞和未决,还有一个函数指针表示处理动作。信号产生时,内核在进程控制块中设置该信号

的未决标志,直到信号递达才清除该标志。在上图的例子中,

1. SIGHUP信号未阻塞也未产生过,当它递达时执行默认处理动作。
2. SIGINT信号产生过,但正在被阻塞,所以暂时不能递达。虽然它的处理动作是忽略,但在没有解除阻塞之前不能忽略这个
信号,因为进程仍有机会改变处理动作之后再解除阻塞。
3. SIGQUIT信号未产生过,一旦产生SIGQUIT信号将被阻塞,它的处理动作是用户自定义函数sighandler。

如果在进程解除对某个信号的阻塞之前,这种信号发生了多次,那么将如何呢?POSIX.1允许系统递送该信号一次或多次,如

果递送多次,则称对这些信号进行排队。Linux是这样实现的:常规信号在递达之前产生多次只计一次,而实时信号在递达之前

产生多次可以依次放在一个队列里。

实践:

#include <stdio.h>
#include <signal.h>
#include <unistd.h>

static void sighandle(int signo){
        printf("receive %d\n",signo);
}

int main(void){
        sigset_t newmask,oldmask;
        sigemptyset(&newmask);
        sigaddset(&newmask,SIGUSR1);
        if(sigprocmask(SIG_BLOCK, &newmask, &oldmask) < 0){
                perror("sigprocmask");
                return -1;
        }

        signal(SIGUSR1, sighandle);
        sleep(20);
        if(sigprocmask(SIG_UNBLOCK, &newmask, &oldmask) < 0){
                perror("sigprocmask");
                return -1;
        }
        while(1){
                pause();
        }
        return 0;
}
运行结果:

root@gmdz-virtual-machine:~# ./a.out &
[1] 7776
root@gmdz-virtual-machine:~# kill -SIGUSR1 7776
root@gmdz-virtual-machine:~# kill -SIGUSR1 7776
root@gmdz-virtual-machine:~# kill -SIGUSR1 7776
root@gmdz-virtual-machine:~# kill -SIGUSR1 7776
root@gmdz-virtual-machine:~# kill -SIGUSR1 7776
root@gmdz-virtual-machine:~# kill -SIGUSR1 7776
root@gmdz-virtual-machine:~# kill -SIGUSR1 7776
root@gmdz-virtual-machine:~# kill -SIGUSR1 7776
root@gmdz-virtual-machine:~# kill -SIGUSR1 7776
root@gmdz-virtual-machine:~# kill -SIGUSR1 7776
root@gmdz-virtual-machine:~# receive 10

一开始sigusr1信号被屏蔽,在sleep的时候,发送多个sigusr1给进程,等进程苏醒后,解除sigusr1的信号屏蔽,此时信号

处理函数只执行了1次,也就是之前发送的信号只计一次。


如果在信号处理函数执行过程中,又有信号产生,会怎样?

使用如下程序进行验证:

#include <stdio.h>
#include <signal.h>

static void sighandle(int signo){
        printf("sig %d start\n",signo);
        sleep(10);
        printf("sig %d end\n",signo);
}

int main(void){
        if(signal(SIGUSR1,sighandle)){
                perror("signal");
                return -1;
        }
        if(signal(SIGUSR2,sighandle)){
                perror("signal");
                return -1;
        }
        while(1){
                pause();
        }
        return 0;
}

1.如果产生的是相同的信号:

root@virtual-machine:~# ./a.out &
[1] 10037
root@virtual-machine:~# kill -SIGUSR1 10037
root@virtual-machine:~# sig 10 start
kill -SIGUSR1 10037
root@virtual-machine:~# kill -SIGUSR1 10037
root@virtual-machine:~# kill -SIGUSR1 10037
root@virtual-machine:~# sig 10 end
sig 10 start
sig 10 end

只有执行完当前的处理函数后才继续执行下一个相同信号的处理函数,因为在执行信号处理函数时,会将该信号屏蔽,在执行完

处理函数后,再解除信号的屏蔽。


2.如果产生的不是相同的信号:

root@virtual-machine:~# ./a.out &
[1] 10140

kill -SIGUSR1 10140    
root@gmdz-virtual-machine:~# sig 10 start
kill -SIGUSR2 10140      //等了7,8秒再执行该语句
sig 12 start
root@gmdz-virtual-machine:~# sig 12 end  //在kill -SIGUSR2 10140执行了10s后出现
sig 10 end   //在sig 12 end出现后立即打印出来

这边的现象就比较奇怪了,如果产生的不是相同的信号,能立即执行新信号的处理函数,但是原来被打断的信号处理函数一直都

没有结束,等到新的信号处理函数结束后,旧信号的处理函数才结束。而且是紧更着新的信号处理函数就打印的。不明白是什么

原因了,等以后再深入地学习后再来搞个明白吧,或者有哪位高人知道的,请告之,谢谢。




相关文章推荐

《UNIX环境高级编程》读书笔记之信号(1)

1.信号的概念 信号时软中断,它提供了一种处理异步时间的方法。 很多条件都会产生信号: (1)用户按某些键时,引发终端产生信号。 (2)硬件异常产生信号:除数0,无效的内存引用等。 (3)进程调用ki...

《UNIX环境高级编程》读书笔记之信号(2)

1.函数sigaction sigaction函数的功能是检查或修改与指定信号相关联的处理动作。其函数原型如下: #inlcude int sigaction(int signo,const ...

UNIX环境高级编程读书笔记(十)—信号 (4)

四、信号屏蔽字: 有时候我们希望进程正确的执行,而不想进程受到信号的影响,比如我们希望上面那个程序在1秒钟之后不结束。这个时候我们就要进行信号的操作了。 信号操作最常用的方法是信号屏蔽。信...

linux信号(二)--unix环境高级编程读书笔记

本文主要介绍了linux中信号操作的一些函数,包括sigemptyset,sigaction,sigsuspend,sigprocmask函数等...

《UNIX环境高级编程》十信号读书笔记

1、信号概念 信号是软件中断,提供了一种处理异步事件的方法。 在头文件《signal.h>中,信号名都被定义为正整数常量(信号编号)。 不存在编号为0的信号,kill函数对信号编号0由特殊的应用。 在...

UNIX环境高级编程读书笔记(十)—信号 (3)

4. 名称:: alarm 功能: set an alarm clock for delivery of a signal ...

《UNIX环境高级编程》笔记--信号集

1.信号集基本操作 我们需要有一个能表示多个信号--信号集(signal set)的数据类型。POSIX.1定义了数据类型sigset_t以包含一个信号 集,并且定义了一下五个处理信号处理信号集函数。...
  • TODD911
  • TODD911
  • 2013年12月08日 19:05
  • 1427

Unix环境高级编程(阅读笔记)----信号集、信号屏蔽函数sigprocmask

信号屏蔽字是指一个进程中当前阻塞而不能够递送给该进程的信号集。 信号集则是一个能表示多个信号的集合的一种数据类型,为sigset_t。 与信号集设置相关的函数有如下几个: //  下列四个函...

UNIX环境高级编程之信号

1、信号本质与来源 信号是在软件层次上对中断机制的一种模拟,在原理上,一个进程收到一个信号与处理器收到一个中断请求可以说是一样的。信号是异步的,一个进程不必通过任何操作来等待信号的到达,事实上,进程...

Unix环境高级编程---信号

参考博客:http://blog.csdn.net/alex_my/article/details/39494129 1. 信号概念  何为信号? 信号是一种软中断,可以由以下情形触...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:《UNIX环境高级编程》笔记--可靠信号
举报原因:
原因补充:

(最多只允许输入30个字)