[学习笔记]信号的高级用法

sigaction函数注册信号处理函数


sigaction函数

q  包含头文件<signal.h>

q  功能:sigaction函数用于改变进程接收到特定信号后的行为。

q  原型:

int  sigaction(int signum,const struct sigaction *act,const struct sigaction *old);

q  参数

q  该函数的第一个参数为信号的值,可以为除SIGKILL及SIGSTOP外的任何一 个特定有效的信号(为这两个信号定义自己的处理函数,将导致信号安装错误)

q  第二个参数是指向结构sigaction的一个实例的指针,在结构 sigaction的实例中,指定了对特定信号的处理,可以为空,进程会以缺省方式对信号处理

q  第三个参数oldact指向的对象用来保存原来对相应信号的处理,可指定oldact为NULL。

q  返回值:函数成功返回0,失败返回-1

signal(num., handle)


sigaction结构体

q  第二个参数最为重要,其中包含了对指定信号的处理、信号所传递的信息、信号处理函数执行过程中应屏蔽掉哪些函数等等

struct sigaction {

void (*sa_handler)(int);   //信号处理程序 不接受额外数据

void (*sa_sigaction)(int, siginfo_t *, void *); //信号处理程序 能接受额外数据,和sigqueue配合使用

sigset_t sa_mask; //

int sa_flags; //影响信号的行为 SA_SIGINFO表示能接受数据

void (*sa_restorer)(void); //废弃

};

注意1:回调函数句柄sa_handler、sa_sigaction只能任选其一。

注意2:The sigaction structure is defined as something like 思考如何测试?


会查找、会用man手册,是通往高手的必经之路。

The siginfo_t parameter to sa_sigaction is a struct with the following elements

siginfo_t {

int      si_signo;  /* Signal number */

int      si_errno;  /* An errno value */

int      si_code;   /* Signal code */

pid_t    si_pid;    /* Sending process ID */

uid_t    si_uid;    /* Real user ID of sending process */

int      si_status; /* Exit value or signal */

clock_t  si_utime;  /* User time consumed */

clock_t  si_stime;  /* System time consumed */

sigval_t si_value;  /* Signal value */

int      si_int;    /* POSIX.1b signal */

void *   si_ptr;    /* POSIX.1b signal */

void *   si_addr;   /* Memory location which caused fault */

int      si_band;   /* Band event */

int      si_fd;     /* File descriptor */

}


实验1sigaction的函数注册信号,基本用法

void handler(int sig)

{

printf("recv a sig=%d\n", sig);

}

__sighandler_t my_signal(int sig, __sighandler_t handler)

{

struct sigaction act;

struct sigaction oldact;

act.sa_handler = handler;

sigemptyset(&act.sa_mask);

act.sa_flags = 0;

if (sigaction(sig, &act, &oldact) < 0)

return SIG_ERR;

return oldact.sa_handler;

}

int main(int argc, char *argv[])

{

struct sigaction act;

sigset_t sa_mask;

act.sa_handler = handler;

act.sa_flags = 0;

sigemptyset(&act.sa_mask);

//测试信号安装函数

//sigaction(SIGINT, &act, NULL);

//模拟signal函数

my_signal(SIGINT, handler);

for (;;)

{

pause();

}

return 0;

}

 


实验2:测试sigaction结构体第三个参数sigset_t sa_mask的作用

/*

struct sigaction {

void (*sa_handler)(int);

void (*sa_sigaction)(int, siginfo_t *, void *);

sigset_t sa_mask;

int sa_flags;

void (*sa_restorer)(void);

} */

//测试sigaction结构体第三个参数sigset_t sa_mask的作用

//作用 sigaddset(&act.sa_mask, SIGQUIT); 加入到sa_mask中的信号,被阻塞(信号处理函数执行的过程中被阻塞)。

//注意:SIGQUIT信号最终还会抵达

int main(int argc, char *argv[])

{

struct sigaction act;

act.sa_handler = handler;

sigemptyset(&act.sa_mask);

sigaddset(&act.sa_mask, SIGQUIT);

act.sa_flags = 0;

if (sigaction(SIGINT, &act, NULL) < 0)

ERR_EXIT("sigaction error");

for (;;)

pause();

return 0;

}

void handler(int sig)

{

printf("recv a sig=%d 信号处理函数执行的时候,阻塞sa_mask中的信号\n", sig);

sleep(5);

}

sigqueue新的信号发送函数


sigqueue函数

q  功能:新的发送信号系统调用,主要是针对实时信号提出的支持信号带有参数,与函数sigaction()配合使用。

q  注意:和kill函数相比Int kill(pid_t pid, int siq)多了参数

q  原型:

         int sigqueue(pid_t pid, int sig, const union sigval value);

q  参数

q   sigqueue的第1个参数是指定接收信号的进程id,第2个参数确定即将发送的信号,第3个参数是一个联合数据结构union sigval,指定了信号传递的参数,即通常所说的4字节值。

q  返回值成功返回0,失败返回-1


q  sigqueue()比kill()传递了更多的附加信息,但sigqueue()只能向一个进程发送信号,而不能发送信号给一个进程组。

q  sigval联合体

typedef union sigval

{

int sival_int;

void *sival_ptr;

}sigval_t;

复制去Google翻译翻译结果

时间: 2024-08-27 10:35:11

[学习笔记]信号的高级用法的相关文章

C#学习笔记之线程 - 高级主题:等待和触发信号

等待信号和触发信号 - Signaling with Wait and Puls 前面讨论事件等待句柄--一个简单的信号机制,一个线程一直阻塞直到接收到另外一个线程的通知. 一个更强大的信号机制被Monitor类所经由静态函数Wait和Pluse(及PulseAll)提供.你自己编写通知逻辑,使用自定义标记和字段(加上锁),然后引入Wait和Pluse命令来阻止自旋.就lock语句和这些函数,你就可以完成AutoResetEvent,ManualResetEven和Semaplore的功能,同样

Python学习笔记八 面向对象高级编程(二)元类

参考教程:廖雪峰官网https://www.liaoxuefeng.com/wiki/0014316089557264a6b348958f449949df42a6d3a2e542c000 在廖老师的学习网站里"使用元类"这部分还是把我给看晕了...网上搜到一篇感觉讲的相对易懂一些,贴出链接:两句话掌握 Python 最难知识点--元类--以此文作为这篇笔记的源本. "道生一,一生二,二生三,三生万物" 1.在Python世界中,"type"即为道

MongoDB学习笔记六:高级操作

[数据库命令]『命令的工作原理』MongoDB中的命令其实是作为一种特殊类型的查询来实现的,这些查询针对$cmd集合来执行.runCommand仅仅是接受命令文档,执行等价查询,因此,> db.runCommand({"drop" : "test"})这个drop调用实际上是这样的: db.$cmd.findOne({"drop" : "test"})当MongoDB服务器得到查询$cmd集合的请求时,会启动一套特殊的逻

《MySQL必知必会学习笔记》:高级联结

创建高级联结 上篇博文简单的介绍了下联结的使用,这篇博文就介绍下如何来创建高级联结. 使用表别名 在前面的学习中,我们知道如何给列取一个别名,现在回顾下,如下: select cust_name,upper(cust_name) as upper_name from customers; 或者是,用concat函数来对客户的姓名和电话进行组合为一个新的信息 ,将这个新的信息取一个别名,如下: 以上就是给列取别名,也可以给计算字段取别名.不仅如此,SQL还允许给表名取别名.这样做有两个主要理由,如

[爬虫学习笔记]MemoryCache缓存的用法学习

      在完成了DNS解析模块之后,我意识到了DNS缓存机制也很有必要.在Redis,Memcache,和.Net自带的Cache之间,考虑到部署问题,最终选择了后者,之前在学习Web及开发的过程中用过System.Web.Caching.Cache这个类库,但是这次的爬虫程序我打算部署为桌面软件,所以选用了System.Runtime.Caching.MemoryCache(后期如有必要也会加入System.Web.Caching.Cache来适配Web端程序).       Memory

[学习笔记]信号发送

kill函数 Kill基本用法 发送信号的函数有kill和raise 区别:kill既可以向自身发送信号,也可以向其他进程发送信号: raise函数向进程自身发送信号. Int kill(pid_t pid, int siq) int raise(int signo) Int kill(pid_t pid, int siq) 参数组合情况解释: kill(pid_t pid, int siq) pid>0 将信号sig发给pid进程 pid=0 将信号sig发给同组进程 pid=-1 将信号si

shell脚本学习笔记之grep命令用法

***grep基本用法 grep [选项] [模式] [文件..] 选项 -c 只输出匹配行的数量 -i 搜索时忽略大小写 -h 查询多文件时不显示文件名 -l 只搜索匹配的文件名,而不列出具体的匹配行 -n 列出所有的匹配行,并显示行号 -s 不显示不存在或无匹配文本的错误信息 -v 显示不包含匹配文本的所有行 -w 匹配整词 -x 匹配整行 -r 递归搜索,不仅所属当前工作目录,而且搜索子目录 -q 禁止储存任何结果,以推迟状态表示所属是否成功 -b 打印匹配行距文件的头部的偏移量,以字节为

shell脚本学习笔记之sed命令用法

sed基本用法 sed:stream editor 行编辑器 sed:模式空间 将匹配的文本内容储存到模式空间中 默认不编辑原文件,仅对模式空间中的数据做处理,而后,处理结束后,架构模式空间的内容显示 sed -n -i:直接修改原文件 -e script -e script:可以同时执行多个脚本 -f file sed -f /scripts  file -r:表示使用扩展正则表达式 sed 'adresscommand' file... -n:静默模式,不再默认显示模式空间中的内容,即只显示

python 学习笔记day07-python函数高级应用

函数高级应用 变量作用域 全局变量 标识符的作用域是定义为其声明在程序里的可应用范围,也就是变量的可见性 在一个模块中最高级别的变量有全局作用域 全局变量的一个特征是除非被删除掉,否则它们的存活到脚本运行结束,且对于所有的函数,他们的值都是可以被访问的 局部变量 局部变量只是暂时地存在,仅仅只依赖于定义他们的函数现阶段是否处于活动 当一个函数调用出现时,某局部变量就进入声明他们的作用域,在那一刻,一个新的局部变量名为那个对象创建了 一旦函数完成,框架被释放,变量将会离开作用域 如果局部与全局有相