Linux--信号阻塞与屏蔽
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--信号阻塞与屏蔽的更多相关文章
- linux信号--阻塞与未决
执行信号的处理动作称为信号递达(Delivery),信号从产生到递达之间的状态,称为信号未决(Pending). 进程可以选择阻塞(Block)某个信号.被阻塞的信号产生时将保持在未决状态,直到进程解 ...
- Linux 信号详解五(信号阻塞,信号未决)
信号在内核中的表示 执行信号的处理动作成为信号递达(Delivery),信号从产生到递达之间的状态称为信号未决(Pending).进程可以选择阻塞(Block)某个信号. 被阻塞的信号产生时将保持在未 ...
- Linux信号-信号集&信号屏蔽字&捕捉信号【转】
转自:https://blog.csdn.net/Lycorisradiata__/article/details/80096203 一. 阻塞信号 1. 信号的常见其他概念 实际执行信号的处理 ...
- linux系统编程之信号(五):信号集操作函数,信号阻塞与未决
一,信号集及相关操作函数 信号集被定义为一种数据类型: typedef struct { unsigned long sig[_NSIG_WORDS]: } sigset_t 信号集用来描述信号的集合 ...
- Linux 信号详解六(可靠信号与不可靠信号)
#include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h&g ...
- Linux信号(signal) 机制分析
Linux信号(signal) 机制分析 [摘要]本文分析了Linux内核对于信号的实现机制和应用层的相关处理.首先介绍了软中断信号的本质及信号的两种不同分类方法尤其是不可靠信号的原理.接着分析了内核 ...
- [置顶] Linux信号相关笔记
最近又温习了一遍Linux中的信号知识,发现有很多东西以前没有注意到,就通过这篇博客记录一下,巩固一下知识点. 一,信号基础: 信号是什么?为了回答这个问题,首先要从异常说起,这里的异常不是指c++/ ...
- 非常好的一篇对linux信号(signal)的解析 (转载)【转】
转自:https://blog.csdn.net/return_cc/article/details/78845346 Linux信号(signal) 机制分析 转载至:https://www.cnb ...
- Linux 信号:signal 与 sigaction
0.Linux下查看支持的信号列表: france@Ubuntux64:~$ kill -l ) SIGHUP ) SIGINT ) SIGQUIT ) SIGILL ) SIGTRAP ) SIGA ...
- Linux信号(signal)机制【转】
转自:http://gityuan.com/2015/12/20/signal/ 信号(signal)是一种软中断,信号机制是进程间通信的一种方式,采用异步通信方式 一.信号类型 Linux系统共定义 ...
随机推荐
- 2018-2019-2-20175225 实验二《Java开发环境的熟悉》实验报告
姓名:张元瑞 学号:20175225 班级:1752 实验课程:JAVA程序设计 实验名称:Java面向对象程序设计 实验时间:2019.4.16 指导老师:娄嘉鹏 实验内容 测试点一 - " ...
- Ext.util.Format.date与Ext.Date.format区别, 转换时间戳
在Extjs中装时间戳使用如下两种都可以: Ext.util.Format.date(time,'U'); Ext.Date.format(time, 'U'); 为了找到它们的区别,查看源代码,以E ...
- Deep Learning系统实训之三:卷积神经网络
边界填充(padding):卷积过程中,越靠近图片中间位置的像素点越容易被卷积计算多次,越靠近边缘的像素点被卷积计算的次数越少,填充就是为了使原来边缘像素点的位置变得相对靠近中部,而我们又不想让填充的 ...
- java 扫描输入
到目前为止,从文件或标准输入读取数据还是一件相当痛苦第事情,一般第解决之道就是读入一行文本,对其进行分词,然后使用Integer Double 等类第各种解析方法来解析数据: //: strings/ ...
- web----Tornado
安装: pip3 install tornado 源码安装 https://pypi.python.org/packages/source/t/tornado/tornado-4.3.tar.gz 简 ...
- bzoj3224 splay板子
开始学习新知识:splay——tree 是个板子题,学习splay可以看博客 https://blog.csdn.net/Clove_unique/article/details/50630280 # ...
- pytest六:parametrize-参数化
pytest.mark.parametrize 装饰器可以实现测试用例参数化. 1.这里是一个实现检查一定的输入和期望输出测试功能的典型例子 import pytest @pytest.mark.pa ...
- python 全栈开发,Day30(纸牌游戏,异常和错误,异常处理)
一.纸牌游戏 ...
- 通过T4模板实现代码自动生成
1:准备.tt模板 using BBFJ.OA.IBLL; using BBFJ.OA.IDAL; using BBFJ.OA.Model; using System; using System.Co ...
- C#4.0特性
C# 4.0的主要主题是动态编程.对象的意义变得越来越“动态”,它们的结构和行为无法通过静态类型来捕获,或者至少编译器在编译程序时无法得知对象的结构和行为. a. 来自动态编程语言——如Python或 ...