signal和sigaction 分析
1:signal 函数
原型: sighandler_t signal(int signum, sighandler_t handler) typedef void (*sighandler_t)(int);
描述:signal函数用来在进程中指定当一个信号到达进程后该做什么处理,主要的两种方式有忽略某些信号,(监听到SIGTERM/SIGINT)退出前的打扫工作。信号处理函数的handler有两个默认值,分别是SIG_IGN和SIG_DFL,表示忽略和默认行为。而且signal函数是阻塞的,比如当进程正在执行SIGUSR1信号的处理函数,此时又来一个SIGUSR1信号,signal会等到当前信号处理函数处理完后才继续处理后来的SIGUSR1,不管后来的多少个SIGUSR1信号,统一看做一个来处理。还有SIGKILL和SIGSTOP这两个信号是signal函数捕捉不到的。
注意:signal函数的行为与linux的版本有关(因为signal属于系统调用),所以移植性不好,可用sigaction函数来替代(sigaction是标准POSIX函数),而且sigaction还扩展了signal函数的功能。
2:sigaction函数
原型:int sigaction(int signum, const struct sigaction *act, struct sigaction *oldact);
参数:
signum:信号值
act:信号的处理参数
oldact:保存信号上次安装时的处理参数(备份的作用)
返回值:0(success),-1(error)
struct sigaction {
void (*sa_handler)(int); //信号处理函数
void (*sa_sigaction)(int, siginfo_t *, void *);
sigset_t sa_mask; //信号屏蔽集
int sa_flags;
void (*sa_restorer)(void);// 已废弃
};
要点:
信号阻塞:同signal函数类似,当正处于某个信号的处理函数中时,这个信号再次到达会被阻塞,待信号处理函数完成之后再处理。
sa_mask :信号屏蔽集,所谓屏蔽并不是忽略,屏蔽的时间段是在信号处理函数执行期间,一旦处理函数执行完毕将会重新唤醒此信号。
sa_flags :SA_RESTART:表示如果一个信号到来时进程正在执行某个系统调用(read/write等),执行完信号处理函数后返回系统调用,此系统调用会重新调用而不会出错。
实例1(sa_mask的使用):
#include <errno.h>
#include <string.h> static void signal_handler(int sig)
{
printf("Got SIGUSR1\n");
printf("Processing SIGUSR1...\n");
sleep();
printf("sleep over\n");
} int main(int argc, char** argv)
{
char buf[];
int n = ;
printf("PID:%d\n", getpid());
struct sigaction act, oldact;
act.sa_handler = signal_handler;
sigemptyset(&act.sa_mask);
sigaddset(&act.sa_mask, SIGINT);//把SIGINT信号加入屏蔽集
//act.sa_flags = SA_RESTART;
if(sigaction(SIGUSR1, &act, &oldact) < )
{
perror("sigaction error");
return -;
}
/*
for(;;){
memset(buf, 0, 20);
if((n = read(0, buf, 20)) < 0){
perror("read:");
}
else{
buf[n] = '\0';
printf("%d type got, string:%s\n", n, buf);
}
}
*/
sleep();
return ;
}
结果:

实例2(sa_flags=SA_RESTART):
#include <stdio.h>
#include <unistd.h>
#include <signal.h>
#include <errno.h>
#include <string.h> static void signal_handler(int sig)
{
printf("Got SIGUSR1\n");
printf("Processing SIGUSR1...\n");
sleep();
printf("sleep over\n");
} int main(int argc, char** argv)
{
char buf[];
int n = ;
printf("PID:%d\n", getpid());
struct sigaction act, oldact;
act.sa_handler = signal_handler;
sigemptyset(&act.sa_mask);
sigaddset(&act.sa_mask, SIGINT);
act.sa_flags = SA_RESTART;//设置SA_RESTART flag
if(sigaction(SIGUSR1, &act, &oldact) < )
{
perror("sigaction error");
return -;
} memset(buf, , );
if((n = read(STDIN_FILENO, buf, )) < ){//行缓冲阻塞
perror("read:");
}
else{
buf[n] = '\0';
printf("%d type got, string:%s\n", n, buf);
} return ;
}
结果:

将 act.sa_flags = SA_RESTART; 这行代码屏蔽之后:

可以看到,SA_RESTART参数确实保证了被中断的系统调用在中断之后恢复之前的状态继续执行(系统调用中断出错应该报告Interrupted system call,不知我这里为何是core dumped?)。
signal和sigaction 分析的更多相关文章
- Linux 信号:signal 与 sigaction
0.Linux下查看支持的信号列表: france@Ubuntux64:~$ kill -l ) SIGHUP ) SIGINT ) SIGQUIT ) SIGILL ) SIGTRAP ) SIGA ...
- Linux进程间通信(一): 信号 signal()、sigaction()
一.什么是信号 用过Windows的我们都知道,当我们无法正常结束一个程序时,可以用任务管理器强制结束这个进程,但这其实是怎么实现的呢?同样的功能在Linux上是通过生成信号和捕获信号来实现的,运行中 ...
- signal() 和 sigaction()
[摘自<Linux/Unix系统编程手册>] Unix系统提供了两种方式来改变信号处置:signal() 和 sigaction(). signal() 的行为在不同Unix实现间存在差异 ...
- 信号处理signal、sigaction、pause、信号嵌套处理、不可重入函数
信号的捕捉和处理 主要由signal和sigaction函数来完成.还有一个函数pause,它可用来响应任何信号,不过不做任何处理. 1.signal函数 typedef void (*sighand ...
- APUE学习笔记——10信号——信号接口函数 signal 和 sigaction
signal函数 signal函数是早起Unix系统的信号接口,早期系统中提供不可靠的信号机制.在后来的分支中,部分系统使用原来的不可靠机制定义signal函数,如 Solaris 10 .而 ...
- linux c 之signal 和sigaction区别
http://blog.csdn.net/muge0913/article/details/7331129 要对一个信号进行处理,就需要给出此信号发生时系统所调用的处理函数.可以对一个特定的信号(除去 ...
- linux 信号signal和sigaction理解
今天看到unp时发现之前对signal到理解实在浅显,今天拿来单独学习讨论下. signal,此函数相对简单一些,给定一个信号,给出信号处理函数则可,当然,函数简单,其功能也相对简单许多,简单给出个函 ...
- Linux信号(signal) 机制分析
Linux信号(signal) 机制分析 [摘要]本文分析了Linux内核对于信号的实现机制和应用层的相关处理.首先介绍了软中断信号的本质及信号的两种不同分类方法尤其是不可靠信号的原理.接着分析了内核 ...
- Linux信号(signal) 机制分析(转)
[摘要]本文分析了Linux内核对于信号的实现机制和应用层的相关处理.首先介绍了软中断信号的本质及信号的两种不同分类方法尤其是不可靠信号的原理.接着分析了内核对于信号的处理流程包括信号的触发/注册/执 ...
随机推荐
- 洛谷【P1052】过河
https://www.luogu.org/problemnew/show/P1052 题目描述 在河上有一座长度为 L 的独木桥, 一只青蛙想沿着独木桥从河的一侧跳到另一侧. 在桥上有一些石子, 青 ...
- PIL包中图像的mode参数
在这里的第一篇. 这篇的是为了说明PIL库中图像的mode参数. 我做的事情是: 在本地找了jpg的图,convert为不同mode,将不同的图截取做了个脑图,有个直观的感觉吧. 把不同mode的图通 ...
- ubuntu 设置全局代理
ubuntu配置shadowsocks全局代理 在mac.window平台下都有shadowsocks客户端,因此这两个平台不叙述太多,现在介绍ubuntu下的配置方法. 1.安装python lin ...
- 第七次ScrumMeeting博客
第七次ScrumMeeting博客 本次会议于11月1日(三)22时整在3公寓725房间召开,持续15分钟. 与会人员:刘畅.辛德泰.窦鑫泽.张安澜.赵奕.方科栋. 1. 每个人的工作(有Issue的 ...
- openresty 安装指南
对于一些常见的 Linux 发行版本,OpenResty 提供 官方预编译包.确保你首先用这种方式来安装. 如果您还没有下载 OpenResty 的源码包, 请到 Download 页下载. 首先,您 ...
- python 动态获取当前运行的类名和函数名的方法
一.使用内置方法和修饰器方法获取类名.函数名 python中获取函数名的情况分为内部.外部,从外部的情况好获取,使用指向函数的对象,然后用__name__属性 复制代码代码如下: def a():pa ...
- 第二次作业——个人项目实战(sudoku)
第二次作业--个人项目实战(sudoku) 一.作业要求地址 第二次作业--个人项目实战 二.Github项目地址 softengineering1--sudoku 三.PSP表格估计耗时 PSP2. ...
- Python @retry装饰器的使用与实现案例(requests请求失败并重复请求)
在爬虫代码的编写中,requests请求网页的时候常常请求失败或错误,一般的操作是各种判断状态和超时,需要多次重试请求,这种情况下,如果想优雅的实现功能,可以学习下retrying包下的retry装饰 ...
- mysql中一些表选项
表选项列表 表选项就是,创建一个表的时候,对该表的整体设定,主要有如下几个: charset = 要使用的字符编码, engine = 要使用的存储引擎(也叫表类型), auto_increment ...
- 【bzoj5147】casino 区间dp
题目描述 赌城拉斯维加斯的米高梅大赌场最近推出了一种新式赌法.它的玩法是由庄家设局(所用赌具是一批五颜六色的筹码),赌徒只要交付一定数额的赌资即可入局.开赌前庄家将手中的筹码依次排开铺成一排构成一局, ...