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系统共定义 ...
随机推荐
- annoy ANN算法 调参
search_k serach_k越大,越准确,但是要在时间和准确率之间取个trade off During the query it will inspect up to search_k node ...
- pytorch实现花朵数据集读取
import os from PIL import Image from torch.utils import data import numpy as np from torchvision imp ...
- JavaScript对象之深度克隆
也不知道从什么时候开始,前端圈冒出了个新词:对象深度克隆.看起来好像很高大上的样子,实际上并不新鲜,在我们的实际项目开发中,你可能早已用到,只不过由于汉字的博大精深,有些原本很简单的事物被一些看似专业 ...
- Laravel Eloquent 自定义返回字段
返回指定字段 Book::select("price", "name")->all(); 返回关系字段关联的属性 Book::select("p ...
- hdu5358 推公式+在一个区间内的尺取+枚举法
尺取+枚举,推出公式以后就是一个枚举加尺取 但是这题的尺取不是对一个值尺取,而是在一个区间内,所以固定左边界,尺取右边界即可 #include<bits/stdc++.h> #define ...
- zoj1716简单的二维树状数组
问一个矩形框在一个大矩形内最多能围几个给定的点 都不用排序,先把所有的点加入树状数组,再直接枚举大矩形的每个格子即可 #include <iostream> #include <st ...
- The last packet sent successfully to the server was 0 milliseconds ago.[nutch---mysql ]
今天在使用JDBC操作mysql时遇到下面的异常信息: 引用 The last packet sent successfully to the server was 0 milliseconds ag ...
- BZOJ1296 [SCOI2009]粉刷匠 动态规划 分组背包
欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ1296 题意概括 有 N 条木板需要被粉刷. 每条木板被分为 M 个格子. 每个格子要被刷成红色或蓝 ...
- Eclipse添加git插件上传项目到github
前提: 在Github已经注册成功自己的账号 新建一个仓库 创建成功后记住url: 首先像安装Pydev一样 点击help的Install New Software 点击Add后添加链接http:// ...
- centos7.2下安装Mysql笔记
centos7.2下安装Mysql笔记 安装 MySQL 适用于 CentOS 7.0 或以后版本: yum install mariadb mariadb-server 适用于 CentOS 6.8 ...