免费排版网站,云南微网站制作哪家好,黄页88网宁波企业名录,华为网站建站问题#xff1a;信号为什么要被保存#xff1f;答#xff1a;信号不会立即处理#xff0c;产生之后#xff0c;处理之前#xff0c;就有时间窗口保存信号#xff0c;必须要把信号保存起来#xff0c;方便后面进行处理。概念#xff1a;1#xff09;实际执行信号的处理…问题信号为什么要被保存答信号不会立即处理产生之后处理之前就有时间窗口保存信号必须要把信号保存起来方便后面进行处理。概念1实际执行信号的处理动作称为信号递达handler2信号从产生到递达之家的状态称为信号未决pending3信号可以选择阻塞block某个信号说人话就是保存该信号但是不进行处理4被阻塞的信号产生时将保持在未决状态直到进程解除对该信号的阻塞才执行递达的动作注意阻塞和忽略不同只要信号被阻塞就不会递达而忽略是递达之后可选的一种处理动作pending 表这张表其实就是一个long long 类型只不过用位图操作来表示普通信号【131】实际是【164】因为还有实时信号是否被保存比特位的位置信号编号比特位的内容是否保存0没有1是block 表这张表和 pending 表一样也是位图操作原来表示普通信号 【131】是否被阻塞比特位的位置信号编号比特位的内容是否被阻塞0没有1是handler 表这张表是函数指针数组专门保存对应的信号编号的处理函数下标 1 信号编号问题signal2myhandle) 底层会做些什么答把我们 myhandle 函数地址根据信号编号 2 来填写到对应的 handler 表里。问题什么叫忽略默认答我们使用 signal 函数时传宏来表示对应的忽略和默认SIG_DFL 和 SIG_IGN 他们两个的本质其实就是 0 和 1 数字只不过把他们强转成函数类型通过对比得出他们是自定义函数函数指针较长还是忽略和默认。问题在信号还没有产生的时候进程就能识别和处理信号了因为程序员已经内置对应管理和处理方法其实就是上面那三张表。结论OS 需要让用户控制信号本质就是访问和操作上面的三张表因为那三张表数据内核结构所以我们要使用系统调用来操作这个三张表其中handler 表系统调用signal blockpending 表这些系统调用函数除了 signal 都是可以获取和设置当前进程的 pending 表和 block 表的。上面的两个函数都有一个参数sigset_t 类型这个内核自定义的类型它其实和pending 和 block 表里的 long long 类型一样都是用来表示信号是否被保存/阻塞通过它可以获取到对应的表其中 sigset_t 称为信号集block表称为 block 信号集、pending 表称为pending 信号集。其中阻塞信号集称为信号屏蔽字Signal Mask)。注意Sigal Mask 类似于 umask 权限掩码。不建议我们用户是直接修改表中的数据而是使用系统调用来修改这些系统调用这里就不再多讲如果想了解可以问一下大模型。sigprocmask 系统调用可以读取或更改进程的信号屏蔽字第一个参数第二个参数传你要修改的屏蔽字第三个参数原来的的屏蔽字防止你要恢复原来的屏蔽字。返回值成功 0 失败-1sigpending 系统调用它可以获取 pending 表参数传一个 pending 表的指针过去。返回值成功0失败-1sigpending 系统调用可以获取 pending 表那么谁来修改他呢答我们用户使用 kill 函数或者命令来让 OS 来修改。#include iostream #include unistd.h #include signal.h void printpending(const sigset_t pending) { for(int signo 31;signo 0;signo--) { if(sigismember(pending,signo))//判断 signo 信号是否存在, { std::cout 1; } else std::cout 0; } std::cout std::endl; } int main() { //屏蔽2号信号 sigset_t block_set,old_set; sigemptyset(block_set);//初始化 sigemptyset(old_set);//初始化 sigaddset(block_set,SIG_SETMASK);//对我们自定义的位图进行屏蔽 2号信号 的操作0 —— 1到这里我们还没有对当前进程的 2号信号进行屏蔽 int n sigprocmask(SIG_SETMASK,block_set,old_set);//修改内核级的 block 表此时已经把 2号信号 屏蔽了 (void)n; std::cout pid: getpid() std::endl; int cnt 1; //获取 pending 表和打印这个表 while(true) { sigset_t pending; sigemptyset(pending);//初始化 n sigpending(pending);//获取 pending 表 printpending(pending); if(cnt 20)//解除对 2号信号的屏蔽 { std::cout 解除对2号信号的屏蔽 std::endl; int n sigprocmask(SIG_SETMASK,old_set,nullptr);//把老的 block 表放回去就相当于解除对2号信号的屏蔽 } cnt; sleep(1); } return 0; }#include iostream #include unistd.h #include signal.h void handl(int signo) { std::cout 处理完成: signo std::endl; } void printpending(const sigset_t pending) { for(int signo 31;signo 0;signo--) { if(sigismember(pending,signo))//判断 signo 信号是否存在, { std::cout 1; } else std::cout 0; } std::cout std::endl; } int main() { signal(2,handl);//更改2号信号的处理函数 //屏蔽2号信号 sigset_t block_set,old_set; sigemptyset(block_set);//初始化 sigemptyset(old_set);//初始化 sigaddset(block_set,SIG_SETMASK);//对我们自定义的位图进行屏蔽 2号信号 的操作0 —— 1到这里我们还没有对当前进程的 2号信号进行屏蔽 int n sigprocmask(SIG_SETMASK,block_set,old_set);//修改内核级的 block 表此时已经把 2号信号 屏蔽了 (void)n; std::cout pid: getpid() std::endl; int cnt 1; //获取 pending 表和打印这个表 while(true) { sigset_t pending; sigemptyset(pending);//初始化 n sigpending(pending);//获取 pending 表 printpending(pending); if(cnt 20)//解除对 2号信号的屏蔽 { std::cout 解除对2号信号的屏蔽 std::endl; int n sigprocmask(SIG_SETMASK,old_set,nullptr);//把老的 block 表放回去就相当于解除对2号信号的屏蔽 } cnt; sleep(1); } return 0; }结论一旦我们解除对某个信号的阻塞该信号就会立即被处理。一旦处理完该函数此时pending 表对应的某个信号的比特位由 1 —— 0问题解除某个信号的屏蔽之后是先进行 pending 表中对应的信号由 1— 0 还是先执行处理函数答先把 pending 表中对应的信号由 1— 0 再执行对应的处理函数。