1)信号是一种软件中断,是在软件层次上对中断的模拟;linux
2)、在平常生活中也有不少信号,好比常见的红绿灯信号,咱们看见红灯就停下,linux中的信号也是相似的,它提供一种机制告诉某个进程在某个时刻该怎样作函数
1)硬件来源:好比咱们按下了键盘或者其它硬件故障;spa
2)软件来源:一些系统函数,最多见的发送信号的函数有kill, raise, alarm和pause;unix
当致使产生信号的事件发生时,内核就产生一个信号。信号产生后,内核一般会在进程表中对进程设置某种形式的标志,当内核设置了这个标志,咱们就说内核向一个进程递送了一个信号code
信号产生和递送之间的时间间隔称为信号未决blog
1)忽略此信号:用signal函数指定SIG_IGN;SIGKILL和SIGSTOP不能被忽略继承
2)捕捉此信号:用signal函数指定一个捕捉函数,指定信号被捕捉到就会调用这个捕捉函数;SIGKILL和SIGSTOP不能被捕捉进程
3)执行系统默认动做:用signal函数指定SIG_DFL事件
做用:设置进程对信号的处理方式(动做)资源
void handler(int signo) { //捕捉函数 } signal(SIGINT,SIG_IGN); signal(SIGINT,SIG_DFL); signal(SIGINT,process);
做用:查看和设置进程对信号的处理方式(动做),比signal()函数多了查看,sigaction函数能传递信息给捕捉函数,signal函数则不能。目前linux中的signal()是经过sigaction()函数实现的。
每一个进程有一个信号屏蔽字,规定了当前要阻塞递送到该进程的信号集(unix提供sigprocmask函数能够得到和更改屏蔽字),对于被阻塞的信号,若是进程对该信号的动做是捕捉或系统默认(即不是忽略),则内核将为该信号保持为未决状态,直到该信号解除阻塞或将对该信号的动做更改成忽略。在这以前,若是这种信号发生了屡次,则发生未决信号排队。
早期unix系统的信号为不可靠信号,它们有下面两点特性(问题):
1)进程对某种信号的处理方式进行设置后(早期的signal函数),第一次接收到这种信号,进程按所设置的方式处理,在这以后,这种信号的处理方式就会被自动重置为系统默认值;
因此当信号的处理方式设成捕捉时,通常会在捕捉函数的开头进行信号处理方式的重建:
1 int sig_int();//捕捉函数 2 . 3 . 4 . 5 signal(SIGINT, sig_int);//设置信号处理方式 6 . 7 . 8 . 9 sig_int() 10 { 11 signal(SIGINT, sig_int);//信号处理方式的重建 12 . 13 . 14 }
可是这种方法也有问题:捕捉到SIGINT,第5行执行,转到第9行调用sig_int(),再进入函数执行第11行,在执行完第5行后到执行第11行前的这段时间里,对SIGINT的处理方式是系统默认值,若是这时发生了SIGINT中断,则程序就终止了。
2)信号可能丢失:对于阻塞信号,不发生未决信号排队,信号阻塞解除后,仅传送该种信号一次,后来的该种信号都将丢失
1)Linux对不可靠信号机制作了改进:信号的处理方式不会被自动重置为系统默认值。所以,Linux下的不可靠信号问题主要指的是信号可能丢失。
2)Linux中信号编号小于SIGRTMIN的信号继承于UNIX系统,是不可靠信号
可靠信号支持未决信号排队,克服了信号可能丢失的问题,linux后来添加的新信号(编号位于SIGRTMIN和SIGRTMAX之间)都是可靠信号
实时信号为可靠信号,非实时信号为不可靠信号
1)UNIX中,SIGCLD在被进程设置成捕捉后,内核会马上检查在这以前是否已经有子进程终止了(等待着wait函数),若是有,则马上调用捕捉函数,而SIGCHLD则不会作这个检查,只管以后的
2)Linux中,SIGCLD等同于SIGCHLD
int kill(pid_t pid, int signo); int raise(int signo);
kill函数将指定信号发送给指定进程(也能够是自身)或进程组,raise函数只能将指定信号发给自身进程。
1)pid>0 将信号传给进程标识码为pid 的进程
2)pid=0 将信号传给和目前进程相同进程组的全部进程
3)pid<0 将信号传给进程组识别码为pid 绝对值的全部进程
4)pid=-1 将信号广播传送给系统内全部的进程
1)Linux指令“kill pid”实际上是向进程标识码为pid 的进程发送默认信号SIGTERM,由于SIGTERM信号编号为15,因此“kill pid”和“kill -s 15 pid”和“kill -15 pid”等价,“-s”能够省略,让“-”加在15上;
2)进程收到SIGTERM信号,将会发生如下事情:进程释放相应资源后终止(相似于让进程正常退出);进程可能仍然继续运行;
3)Linux指令“kill -s 9 pid”表示向进程发送SIGKILL信号来强制杀死进程;
4)SIGTERM信号能够被捕捉和忽略,SIGKILL信号则不能够;
5)Linux信号编号为0的信号为空信号,“kill -s 0 pid”常被用来肯定一个指定pid的进程是否存在;
unsigned int alarm(unsigned int seconds); int pause(void);
1)alarm函数设置一个定时器,通过seconds秒定时器超时,会产生SIGALRM信号,此信号的默认动做是终止进程,通常将信号设置成捕捉,就能够实现相应的定时操做了;
2)pause函数是进程挂起直到捕捉到某个信号,与alarm函数配合时只有当捕捉函数执行完,pause函数才释放进程;
3)每一个进程只能有一个闹钟时间,若是在调用alarm时,以前已经为该进程注册的定时器尚未超时,则这个定时器的剩余时间将会做为本次调用alarm的返回值,而后闹钟时间被更新:
ret = alarm(5); sleep(3); ret = alarm(5); //ret值为2,闹钟时间从新定成5s sleep(1); ret = alarm(5); //ret值为4,闹钟时间从新定成5s
15.常见信号
1) SIGCHLD:一个进程终止时,SIGCHLD信号被送给其父进程
2)SIGINT:中断信号,ctrl+c,用来中断进程
3)SIGKILL(9):强制杀死任一进程,不可被捕捉和忽略
4)SIGSTOP:一个做业控制信号,中止一个进程,不可被捕捉和忽略
5)SIGTERM(15):kill命令默认的终止信号
6)SIGPIPE:在读进程已终止时进行写操做,将产生此信号