软件条件产生信号
alarm函数
说明:设置定时器(闹钟)。在指定多少秒后,内核会给当前进程发送14)SIGALRM信号,进程收到该信号,默认动作终止进程。无论进程处于某种状态都会记时
原型:
unsigned int alarm(unsigned int seconds);
返回0或剩余的秒数
取消定时器:alarm(0)
查看当前进程执行的秒数:time 可执行程序
实际执行时间=系统时间+用户时间+等待时间(绝大多数时间都是在等待中度过)
程序运行的瓶颈在于IO,优化程序,首先优化IO 例如: 往屏幕上打印,很浪费时间。我们可以重定向往文件中输出 |
分别向文件和终端打印,观察1s可以计算多少个数:
#include <iostream> #include <sys/types.h> #include <unistd.h> using namespace std; int main(void) { int i=0; alarm(1); //定时1秒,14)SIGALRM 终止进程 while(1) { ++i; //cout<<i<<endl; printf("%d\n",i); //printf比cout更快一点点 } return 0; }
#include <iostream> #include <sys/types.h> #include <unistd.h> #include <sys/stat.h> #include <fcntl.h> using namespace std; int main(void) { int i=0; int fd = open("./a.txt",O_RDWR|O_CREAT,0777); dup2(fd,STDOUT_FILENO); alarm(1); //定时1秒,14)SIGALRM 终止进程 while(1) { ++i; //cout<<i<<endl; printf("%d\n",i); //printf比cout更快一点点 } return 0; }
大概71倍
setitimer函数
说明:设置定时器(闹钟)。可替代alarm函数,精度微妙us,可实现周期定时
原型:int setitimer(int which,const struct itimerval *new_value,struct itimerval *old_value);
成功返回 0 失败-1,并设置error
参数:
which:指定定时方式
1.自然定时:ITIMER_REAL --> 14)SIGARM
2.虚拟空间计时(用户空间):ITIMER_VIRTUAL --> 26)SIGVTALRM 占用cpu的时间
3.运行时间(用户+内存):ITIMER_PROT --> 27)SIGPROF cpu+系统调用
old_value
上一次定时的时间
it_interval:两次触发定时器的间隔时间
it_value:定时器定时的时间
如果都为0,表示清0
案例1:
使用setitemer函数实现alarm,重复计算1s
#include <iostream> #include <sys/time.h> using namespace std; void myalarm(unsigned long sec) { itimerval it,oldit; it.it_interval.tv_sec = 0; it.it_interval.tv_usec = 0; it.it_value.tv_sec = 1; it.it_value.tv_usec = 0; int ret = setitimer(ITIMER_REAL,&it,&oldit); if(ret == -1) { perror("setitimer err"); exit(-1); } return; } int main(void) { int i; myalarm(1); while(1) { printf("%d\n",i); ++i; } return 0; }
案例2:
#include <iostream> #include <sys/time.h> #include <signal.h> void func(int signo) { printf("wo shi da sha bi\n"); //raise(SIGKILL); } int main(void) { signal(SIGALRM,func); //信号捕捉,当SIGALRM递达,则执行func itimerval it,oldit; it.it_interval.tv_sec = 2; it.it_interval.tv_usec = 0; it.it_value.tv_sec = 1; it.it_value.tv_usec = 0; if(setitimer(ITIMER_REAL,&it,&oldit) == -1) { perror("setitimer err"); exit(-1); } while (1) { /* code */ } return 0; }
信号集操作函数
内核通过读取未决信号集来判断信号是否应该被处理。信号屏蔽字mask可以影响未决信号集。而我们可以在应用程序中自定义set来改变mask。以达到屏蔽指定信号的目的。
信号集设定
sigset_t set; //unsigned long (8字节 64bit)
int sigemptyset(sigset_t * set); //将某个信号集清0
int sigfillset(sigset_t *set); //将某个信号集置1
int sigaddset(sigset_t *set,int signum); //将某个信号加入信号集
int sigdelset(sigset_t *set,int signum); //将某个信号清除信号集
int sigismember(const sigset_t *set,int signum); //判断某个信号是否在信号集中
sigprocmask函数
int sigprocmask(int how,const sigset_t *set,sigset_t *oldset);
成功:0 失败:-1,设置error
参数:
how:假设当前信号屏蔽字为 mask
1.SIG_BLOCK:mask = mask | set
2.SIG_UNBLOCK: mask = mask | ~set
3.SIG_SETMASK: mask = set
sigpending函数
读取当前进程的未决信号集
int sigpending(sigset_t *set);
成功:0 失败:-1,设置error
#include <iostream> #include <sys/time.h> #include <signal.h> #include <unistd.h> void printset(sigset_t *ped) { for(int i=1;i<32;++i) { if(sigismember(ped,i) == 1) { putchar('1'); } else { putchar('0'); } } } int main(void) { sigset_t set,ped; sigemptyset(&set); sigaddset(&set,SIGINT); sigaddset(&set,SIGQUIT); sigfillset(&set); sigprocmask(SIG_BLOCK,&set,NULL); while(1) { sigpending(&ped); printset(&ped); sleep(1); } return 0; }
信号捕捉
signal函数
注册一个信号捕捉函数
typedef void(*sighandler_t)(int);
sighandler_t signal(int signum,.sighandler handler);
handler:
SIG_IGN 忽略此信号
SIG_DFL 使用信号默认的行为
#include <iostream> #include <signal.h> using namespace std; typedef void (*sighandler_t)(int); void catchsignal(int signo) { cout<<"aaa"<<endl; } int main(void) { sighandler_t headler; headler = signal(SIGINT,catchsignal); if(headler == SIG_ERR) { perror("dignal err"); exit(-1); } while (1) { /* code */ } return 0; }
sigaction函数
int sigaction(int signum,const struct sigaction *act,struct sigaction *oldact);
成功:0 失败:-1,并设置error
act:新的处理方式
oldact:传出参数,旧的处理方式
重要的参数:
sa_handler:指定信号捕捉后的处理函数名。(也可以指定SIG_IGN SIG_DFL)
sa_mask:调用信号处理函数时,所要屏蔽的信号集合(阻塞信号集)
sa_flags:通常设置为0,表示默认属性(对正在处理的信息设置屏蔽)
信号捕捉特性
1.进程正常运行时,默认在PCB中有一个阻塞信号集和未决信号集。对于阻塞信号集,它决定自动屏蔽哪些信号。当注册了某个信号捕捉函数,捕捉到信号以后,要调用该函数。而该函数有可能执行时间很长,在这期间所需要屏蔽的信号由sa_mask来决定。当信号处理完后,又恢复为PCB中的阻塞信号集决定
2.xxx信号捕捉函数执行期间,xxx信号自动被屏蔽
3.阻塞的常规信号不支持排队,产生多次只记录一次
信号捕捉函数案例
1.自定义一个信号捕捉函数
#include <iostream> #include <signal.h> #include <unistd.h> using namespace std; void headler(int signo) { cout<<"我是你大爷..."<<endl; sleep(2); return; } int main(void) { struct sigaction act; act.sa_handler = headler; act.sa_flags = 0; sigemptyset(&act.sa_mask); sigaddset(&act.sa_mask,SIGQUIT); int ret = sigaction(SIGINT,&act,NULL); if(ret == -1) { perror("sigaction err"); exit(-1); } while(1) { } return 0; }
2.验证在信号处理函数执行期间,该信号多次递送,那么只在处理函数支行技术和,处理一次
#include <iostream> #include <signal.h> #include <unistd.h> using namespace std; void headler(int signo) { cout<<"我是你大爷..."<<endl; sleep(2); cout<<"-----end-------"<<endl; return; } int main(void) { struct sigaction act,old; act.sa_handler = headler; act.sa_flags = 0; sigemptyset(&act.sa_mask); sigaction(SIGINT,&act,&old); while(1) { } sigaction(SIGINT,&old,NULL); return 0; }
3.验证sa_mask的屏蔽作用
#include <iostream> #include <signal.h> #include <unistd.h> using namespace std; void headler(int signo) { cout<<"我是你大爷..."<<endl; sleep(2); return; } int main(void) { struct sigaction act; act.sa_handler = headler; act.sa_flags = 0; sigemptyset(&act.sa_mask); sigaddset(&act.sa_mask,SIGQUIT); int ret = sigaction(SIGINT,&act,NULL); if(ret == -1) { perror("sigaction err"); exit(-1); } while(1) { } return 0; }
内核实现信号捕捉过程
信号的产生和处理都是由内核进行的
频繁的在用户态和内核态进行切换是很浪费时间的
时态竞争
当前进程在执行期间,由于竞争的原因,导致程序执行的时序在先后两次的执行中有不同的结果
比如A准备睡觉,定了一个闹钟(定时器),在10s后叫醒他。
由于时序可能会有两种情况:
正常:10s后A被闹钟叫醒
异常:A在5s时被B叫醒,并且A跟着B一起去打牌,打了20s,那么在打牌的期间,
闹钟正常响起,但是并没有叫醒A
时序问题分析:
假设我们想要通过pause和alarm实现sleep:
1.注册SIGALARM信号处理函数(sigaction...)
2.调用alarm(1)函数设定闹钟为1秒
3.函数调用结束,开始计时倒数。正在这时,进程失去CPU(时间片轮转,进程调度算法决定),该进程处于就绪状态,等待CPU
4.但是此时时自然计时,倒数依旧在进行着,如果此时倒数结束,alarm发送信号,信号将递达,但是此时进程处于挂起状态,每一办法通过信号捕捉函数进行处理,所有处于未决状态。
5.当进程再次获得CPU,SIGALARM信号递达,执行处理函数函数
6.信号处理函数结束,回到主控制程序,执行pause(),进程将会被挂起,等待alarm唤醒,可以再没有人来唤醒他了
#include <iostream> using namespace std; void myheadler(int signo) { cout<<"aa"<<endl; } int main(void) { signal(SIGALARM,myheadler); alarm(1); pause(); while(1); return 0; }
解决时序问题
可以通过设置屏蔽SIGALRM的方法来控制程序执行逻辑,但无论如何设置,程序都有可能在“解除信号屏蔽字”与“挂起等待信号”这个两个操作间隙失去CPU。除非将这两个步骤合成原子操作。sigsuspend可以实现。
int sigsuspend(const sigset_t *mask) 挂起等待信号
sigsuspend函数调用期间,进程信号屏蔽字由参数mask决定
可将某个信号从临时信号屏蔽集种删除,这样在调用sigsuspend时将解除对该信号的屏蔽,然后挂起等待,当sigsuspend返回时,信号屏蔽字恢复原来的值。
#include <iostream> using namespace std; void myheadler(int signo) { cout<<"aa"<<endl; } int main(void) { signal(SIGALARM,myheadler); sigset_t mask; sigfillset(&mask); sigdelset(&mask,SIGALARM); alarm(1); sigsuspend(&mask); while(1); return 0; }
时序问题总结
竞态条件,跟系统负载有很紧密的关系,体现除信号的不可靠性。系统负载越严重,信号不可靠性越强。不可靠由其实现原理所致。信号是通过软件方式实现,每次系统调用结束后或中断处理结束后,需通过扫描PCB中的未决信号集来判断是否应处理某个信号。当系统负载过重时,会出现混乱
可不可重入函数
一个函数在被调用执行期间(尚未调用结束),由于某时序又被重复调用,称之为"重入"。
1.定义可重入函数,函数内不能包含全局变量集static变量,不能使用malloc、free
2.信号捕捉函数应设计为可重入函数
3.不可重入的原理:使用静态数据结构 调用了malloc和new (不是栈结构) 是标准I/O
SIGCHLD信号
当子进程停止或者结束时,会向父进程发送SIGCHLD信号,该信号默认处理动作是忽略,所有产生僵尸进程。
我们之前的程序是,在父进程中调用 wait或者waitpid来回收子进程,此时父进程要么阻塞等待,要么非阻塞而采用轮询的方式。那么将会导致父进程不能做其他工作,只能等着回收子进程。
我们可以利用SIGCHLD信号来捕捉,当有SIGCHLD信号产生时,父进程执行信号捕捉函数。
我们先来看一段程序:
#include <iostream> #include <signal.h> #include <unistd.h> #include <wait.h> using namespace std; void sigchldHeadler(int signo) { pid_t pid; int status; while((pid=waitpid(-1,&status,WNOHANG))>0) { printf("回收成功:%d ok\n",pid); if(WIFEXITED(status)) { printf("退出状态:%d\n",WEXITSTATUS(status)); } if(WIFSIGNALED(status)) { printf("退出状态(信号):%d\n",WTERMSIG(status)); } } } int main(void) { //创建10个子进程 int i; for(i=0;i<10;i++) { pid_t pid = fork(); if(pid == 0) { break; } } if(i < 10) { printf("I am child:%d\n",getpid()); sleep(1); } else if(i == 10) { struct sigaction act; act.sa_handler = sigchldHeadler; act.sa_flags = 0; sigemptyset(&act.sa_mask); sigaction(SIGCHLD,&act,NULL); while(1) { printf("I am parent:%d\n",getpid()); sleep(1); } } return 0; }
我们来思考几个问题:
当父进程在执行信号捕捉函数时,又有子进程死亡。或者有多个子进程死亡。我们知道信号集是位图机制,是不支持排队的。
当父进程的信号捕捉函数还没有注册,就已经有子进程结束了。导致僵尸进程...
或者在fork之前将SIGCHLD信号设置屏蔽
完整:
#include <iostream> #include <signal.h> #include <unistd.h> #include <wait.h> using namespace std; void sigchldHeadler(int signo) { pid_t pid; int status; while((pid=waitpid(-1,&status,WNOHANG))>0) { printf("回收成功:%d ok\n",pid); if(WIFEXITED(status)) { printf("退出状态:%d\n",WEXITSTATUS(status)); } if(WIFSIGNALED(status)) { printf("退出状态(信号):%d\n",WTERMSIG(status)); } } } int main(void) { //设置阻塞 sigset_t set; sigemptyset(&set); sigdelset(&set,SIGCHLD); sigprocmask(SIG_BLOCK,&set,NULL); //创建10个子进程 int i; for(i=0;i<10;i++) { pid_t pid = fork(); if(pid == 0) { break; } } if(i < 10) { printf("I am child:%d\n",getpid()); //sleep(1); } else if(i == 10) { struct sigaction act; act.sa_handler = sigchldHeadler; act.sa_flags = 0; sigemptyset(&act.sa_mask); sigaction(SIGCHLD,&act,NULL); while(1) { printf("I am parent:%d\n",getpid()); sleep(1); } } return 0; }
信号传参
前面讲到信号是不能携带大量数据的,一般通过 kill 来发送信号
但是信号可以携带数据,可以携带少量。
通过sigqueue函数,可在向指定进程发送信号的同时携带参数
int sigqueue(pid_t pid,int sig,const union sigval value);
成功返回0,失败-1,设置error
注意事项:
向指定进程发送指定信号的同时携带数据。不能够传地址,不能进程之间虚拟地址空间各自独立,当前进程地址传递给另一进程没有实际意义
捕捉函数:
int sigaction(int signum,const struct sigaction *act,struct sigaction *oldact);
成功:0 失败:-1,并设置error
act:新的处理方式
oldact:传出参数,旧的处理方式
不使用sa_handler,而使用sa_sigaction。sa_flags必须指定为SA_SIGINFO
中断系统调用
系统调用分为两类:
慢速系统调用:可能导致进程永远阻塞,如果在阻塞期收到一个信号,该系统调用就会被断开,不再继续执行,也可以设定系统调用重启(read、write、pause、wait...)
其他系统调用:getpid,getpid、fork
sa_flags可以设置为 SA_RESTART