Chinaunix首页 | 论坛 | 博客
  • 博客访问: 2172505
  • 博文数量: 361
  • 博客积分: 10828
  • 博客等级: 上将
  • 技术积分: 4161
  • 用 户 组: 普通用户
  • 注册时间: 2010-01-20 14:34
文章分类

全部博文(361)

文章存档

2011年(132)

2010年(229)

分类: C/C++

2010-04-12 19:25:28

1. 概念:

信号屏蔽字就是进程中被阻塞的信号集, 这些信号不能发送给该进程, 它们在该进程中被"屏蔽"了. 后面我们会提到, 实际上它们是被阻塞了.


2. 信号屏蔽函数:

#include

int sigprocmask(int how, const sigset_t *restrict set, sigset_t *restrict oset);

成功则返回0, 出错则返回-1.
sigprocmask函数有3个参数:

how: 修改信号屏蔽字的方式.
set: 把这个信号集设为新的当前信号屏蔽字. 如果为NULL则不改变.
oset: 保存进程旧的信号屏蔽字. 如果为NULL则不保存.
参数中的how可以取3个值:

sigprocmask中的how参数 how 说明
SIG_BLOCK 修改后, 该进程新的信号屏蔽字是其当前屏蔽字和set指向的信号集的并集.
SIG_UNBLOCK 修改后, 该进程新的信号屏蔽字是其当前屏蔽字和set指向的信号集的补集的交集.
SIG_SETMASK 修改后, 该进程新的信号屏蔽字将被set指向的信号集的值代替


 另外要说的是, sigprocmask只为单线程定义的, 在多线程中要使用pthread_sigmask.


3. 未处理的信号:

在调用信号屏蔽的相关函数后, 被屏蔽的信号对于调用进程是阻塞的, 不能发送给调用进程, 因此是未决的. 取得这些阻塞的信号集, 可以通过调用sigpending函数.

#include

int sigpending(sigset_t *set);

成功则返回0, 出错则返回-1.

4. 实例:

下面通过一个简单的实例来说明这篇文章中所讲到的两个函数.

#include
#include
#include
#include

/* SIGQUIT handler */
static void sig_quit(int signo)
{
    printf("SIGQUIT is caught ");
}

int main()
{
    sigset_t new, old, pend;

    /* Handle SIGQUIT */
    if (signal(SIGQUIT, sig_quit) == SIG_ERR)
    {
        perror("signal");
        exit(1);
    }

    /* Add SIGQUIT to sigset */
    if (sigemptyset(&new) < 0)
        perror("sigemptyset");
    if (sigaddset(&new, SIGQUIT) < 0)
        perror("sigaddset");

    /* Mask SIGQUIT */
    if (sigprocmask(SIG_SETMASK, &new, &old) < 0)
    {
        perror("sigprocmask");
        exit(1);
    }

    printf("SIGQUIT is blocked ");
    printf("Now try Ctrl \  ");

    sleep(5); /* SIGQUIT will pending */

    /* Get pending */
    if (sigpending(&pend) < 0)
        perror("sigpending");

    if (sigismember(&pend, SIGQUIT))
        printf("SIGQUIT pending ");

    /* Restore signal mask */
    if (sigprocmask(SIG_SETMASK, &old, NULL) < 0)
    {
        perror("sigprocmask");
        exit(1);
    }

    printf(" SIGQUIT unblocked ");
    printf("Now try Ctrl \  ");

    sleep(5);

    return 0;
}

这个程序在开始的时候用sigprocmask屏蔽了SIGQUIT(ctrl+\触发), 在5秒内触发的该信号将可以从sigpending中获得; 然后程序把SIGQUIT解除屏蔽(恢复以前的屏蔽字), 此时再触发该信号将调用sig_quit信号处理函数.

 

阅读(4921) | 评论(0) | 转发(2) |
0

上一篇:信号集

下一篇:sigaction

给主人留下些什么吧!~~