1. sigprocmask函数提供屏蔽和解除屏蔽信号的功能。 
从而实现关键代码的运行不被打断。 
函数声明如下:

int sigprocmask(int how, const sigset_t *set, sigset_t *oldset);
其中参数 how可设置的参数为:SIG_BLOCK, SIG_UNBLOCK,SIG_SETMASK 
SIG_BLOCK: 
按照参数 set 提供的屏蔽字,屏蔽信号。并将原信号屏蔽保存到oldset中。 
SIG_UNBLOCK: 
按照参数 set 提供的屏蔽字进行信号的解除屏蔽。针对Set中的信号进行解屏。 
SIG_SETMASK: 
按照参数 set 提供的信号设置重新设置系统信号设置。

2. 信号屏蔽与解屏常见实现 
方法一: SIG_BLOCK, SIG_UNBLOCK成对实现 
优点oldset可以不管。

方法二: 
SIG_BLOCK设置屏蔽,保存原有信号设置。 
SIG_SETMASK重新恢复原有设置。

3. 屏蔽过程中接受到的信号如何处理 
在信号屏蔽过程中,出现的所有被屏蔽的信号,不管发生多少次,在信号解除屏蔽后,系统会执行一次被屏蔽信号上的操作。

#include<stdio.h>
#include<signal.h>
#include<unistd.h> int flag_sigusr1 = 0;
int flag_sigusr2 = 0; void sig_usr1(int signo)
{
fprintf(stdout, "caught SIGUSR1\n");
flag_sigusr1 = 1;
return;
} void sig_usr2(int signo)
{
fprintf(stdout, "caught SIGUSR2\n");
flag_sigusr2 = 1;
return;
} int main(void){
sigset_t newmask, oldmask;
signal(SIGUSR1, sig_usr1);
signal(SIGUSR2, sig_usr2); fprintf(stdout, "catch sigusr1 can break\n"); while(1)
{
if(flag_sigusr1)
{
fprintf(stdout, "break\n");
break;
}
sleep(5);
}
fprintf(stdout, "first while was broken\n"); //重新设置为0
flag_sigusr1 = 0;
flag_sigusr2 = 0; // block SIGUSR1
sigemptyset(&newmask);
sigaddset(&newmask, SIGUSR1);
if(sigprocmask(SIG_BLOCK, &newmask, &oldmask) < 0)
{
perror("sigprocmask error");
} fprintf(stdout, "only catch sigusr2 can break, because sigusr1 has been blocked\n");
while(1)
{
if(flag_sigusr1 || flag_sigusr2)
{
fprintf(stdout, "break\n");
break;
}
sleep(5);
}
fprintf(stdout, "second while was broken\n"); fprintf(stdout, "after second while was broken, flag_sigusr1=%d, flag_sigusr2=%d\n", flag_sigusr1, flag_sigusr2); return 0;
}

  

多线程情况下每个线程共用信号处理函数,但是每个线程可以选择自己是否block某个信号。

再看一个多线程的例子:子线程的功能同上,主线程接收到hup信号会向子线程发送usr2信号。

#include<stdio.h>
#include<signal.h>
#include<unistd.h>
#include<pthread.h> int flag_sigusr1 = 0;
int flag_sigusr2 = 0;
int flag_sighup = 0; void sig_usr1(int signo)
{
fprintf(stdout, "sig|caught SIGUSR1\n");
flag_sigusr1 = 1;
return;
} void sig_usr2(int signo)
{
fprintf(stdout, "sig|caught SIGUSR2\n");
flag_sigusr2 = 1;
return;
} void sig_hup(int signo)
{
fprintf(stdout, "sig|caught SIGHUP\n");
flag_sighup = 1;
return;
} void *thread_control_signal(void *arg)
{
sigset_t newmask, oldmask;
sigemptyset(&newmask); //thread block sighup
sigemptyset(&newmask);
sigaddset(&newmask, SIGHUP);
if(pthread_sigmask(SIG_BLOCK, &newmask, &oldmask) < 0)
{
perror("sigprocmask error");
} fprintf(stdout, "thread|first while. catch sigusr1 or sigusr2 can break\n");
while(1)
{
if(flag_sigusr1 || flag_sigusr2)
{
fprintf(stdout, "thread|break\n");
break;
}
sleep(5);
}
flag_sigusr1 = 0; //thread block SIGUSR1
sigaddset(&newmask, SIGUSR1);
if(pthread_sigmask(SIG_BLOCK, &newmask, &oldmask) < 0)
{
perror("sigprocmask error");
} fprintf(stdout, "thread|first while. catch sigusr2 can break\n");
while(1)
{
if(flag_sigusr1 || flag_sigusr2)
{
fprintf(stdout, "break\n");
break;
}
sleep(10);
}
fprintf(stdout, "thread|thread exit\n");
return (void *)0;
} int main()
{
sigset_t newmask;
pthread_t tid;
int signo; //signal action
signal(SIGUSR1, sig_usr1);
signal(SIGUSR2, sig_usr2);
signal(SIGHUP , sig_hup); if(pthread_create(&tid, NULL, thread_control_signal, NULL) < 0)
{
perror("create pthread failed");
return -1;
} //main thread block sigusr1
sigemptyset(&newmask);
sigaddset(&newmask, SIGUSR1);
if(pthread_sigmask(SIG_BLOCK, &newmask, NULL) < 0)
{
perror("sigprocmask error");
} //main thread wait sighup
sigemptyset(&newmask);
sigaddset(&newmask, SIGHUP);
if(sigwait(&newmask, &signo) < 0)
{
perror("sigwait failed");
return -1;
}
fprintf(stdout, "main|get SIGHUP\n"); pthread_kill(tid, SIGUSR2);
pthread_kill(tid, SIGUSR2);
pthread_join(tid, NULL); fprintf(stdout, "main|exit\n");
return 0;
}

  

kill函数向进程发送信号,pthread_kill用于向线程发送信号。

Linux--信号阻塞与屏蔽的更多相关文章

  1. linux信号--阻塞与未决

    执行信号的处理动作称为信号递达(Delivery),信号从产生到递达之间的状态,称为信号未决(Pending). 进程可以选择阻塞(Block)某个信号.被阻塞的信号产生时将保持在未决状态,直到进程解 ...

  2. Linux 信号详解五(信号阻塞,信号未决)

    信号在内核中的表示 执行信号的处理动作成为信号递达(Delivery),信号从产生到递达之间的状态称为信号未决(Pending).进程可以选择阻塞(Block)某个信号. 被阻塞的信号产生时将保持在未 ...

  3. Linux信号-信号集&信号屏蔽字&捕捉信号【转】

    转自:https://blog.csdn.net/Lycorisradiata__/article/details/80096203 一. 阻塞信号 1. 信号的常见其他概念    实际执行信号的处理 ...

  4. linux系统编程之信号(五):信号集操作函数,信号阻塞与未决

    一,信号集及相关操作函数 信号集被定义为一种数据类型: typedef struct { unsigned long sig[_NSIG_WORDS]: } sigset_t 信号集用来描述信号的集合 ...

  5. Linux 信号详解六(可靠信号与不可靠信号)

    #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h&g ...

  6. Linux信号(signal) 机制分析

    Linux信号(signal) 机制分析 [摘要]本文分析了Linux内核对于信号的实现机制和应用层的相关处理.首先介绍了软中断信号的本质及信号的两种不同分类方法尤其是不可靠信号的原理.接着分析了内核 ...

  7. [置顶] Linux信号相关笔记

    最近又温习了一遍Linux中的信号知识,发现有很多东西以前没有注意到,就通过这篇博客记录一下,巩固一下知识点. 一,信号基础: 信号是什么?为了回答这个问题,首先要从异常说起,这里的异常不是指c++/ ...

  8. 非常好的一篇对linux信号(signal)的解析 (转载)【转】

    转自:https://blog.csdn.net/return_cc/article/details/78845346 Linux信号(signal) 机制分析 转载至:https://www.cnb ...

  9. Linux 信号:signal 与 sigaction

    0.Linux下查看支持的信号列表: france@Ubuntux64:~$ kill -l ) SIGHUP ) SIGINT ) SIGQUIT ) SIGILL ) SIGTRAP ) SIGA ...

  10. Linux信号(signal)机制【转】

    转自:http://gityuan.com/2015/12/20/signal/ 信号(signal)是一种软中断,信号机制是进程间通信的一种方式,采用异步通信方式 一.信号类型 Linux系统共定义 ...

随机推荐

  1. Linux服务器性能评估

    一.影响Linux服务器性能的因素 1. 操作系统级 CPU 内存 磁盘I/O带宽 网络I/O带宽 2. 程序应用级 二.系统性能评估标准 影响性能因素 影响性能因素 评判标准 好 坏 糟糕 CPU ...

  2. X-Forwarded-For 的一些理解

    X-Forwarded-For 是一个 HTTP 扩展头部, 主要是为了让 Web 服务器获取访问用户的真实 IP 地址(其实这个真实未必是真实的,后面会说到). 那为什么 Web 服务器只有通过 X ...

  3. PYTHON-基本数据类型-元祖类型,字典类型,集合类型

    内容: 1. 元组 2. 字典 3. 集合=========================== 元祖类型什么是元组: 元组就是一个不可变的列表============================ ...

  4. css怎么让页面上的内容不能被选中

    body{     -webkit-user-select:none;     -moz-user-select:none;     -ms-user-select:none;     user-se ...

  5. 步步为营-33-Md5(32)加密与Base64加密

    说明: 1:直接贴码 using System; using System.Collections.Generic; using System.ComponentModel; using System ...

  6. 指定一个M3U8文件,判断它包含的TS文件是不是都存在。指定一个Office生成的Swf文件,判断它包含的Swf文件是不是完整都存在。

    static void Main(string[] args) { //检查M3u8文件 var fiPath = @"D:\Work\CloudPlatformUtil\CloudPlat ...

  7. python yield,到这个层次,才能叫深入哈

    http://python.jobbole.com/88677/?utm_source=blog.jobbole.com&utm_medium=relatedPosts ~~~~~~~~~~~ ...

  8. 点分治 poj1741

    题意: 给出一颗树,询问有多少对点对距离<=k 链接: http://poj.org/problem?id=1741 题解: 点分治的模板题 点分治即采用分治思想分而治之 考虑一颗子树内距离&l ...

  9. 解决python中遇到的乱码问题

    1. 解决中文乱码的一种可行方法 # -*- coding:utf-8 -*- from __future__ import unicode_literals import chardet def s ...

  10. 【Java】 剑指offer(64) 求1+2+…+n

      本文参考自<剑指offer>一书,代码采用Java语言. 更多:<剑指Offer>Java实现合集   题目 求1+2+…+n,要求不能使用乘除法.for.while.if ...