信号(singal)机制是Unix系统中最为古老的进程间通信机制,很多条件能够产生一个信号:
1 当用户按下某一按键时,产生信号。
2 硬件异常产生信号:除数为0、无效的存储访问等等。这些情况往往由硬件检测到,将其通知内核,然后内核产生适当的信号通知进程,例如,内核对正访问一个无效存储区的进程产生一个SIGSEGV信号。
3 进程可以使用Kill函数将信号发送给另外一个信号。
4 进程可用Kill命令将信号发送给其他进程。
下面是几种常见的信号:
SIGHUP:从终端发出的结束信号
SIGINT: 来自键盘的中断信号(ctrl+c)
SIGKILL:该信号结束接收信号的进程
SIGTERM:kill命令发出的信号
SIGCHLD:标识子进程停止或结束的信号
SIGSTOP:来自键盘(ctrl+z)或调试程序的停止执行信号
当信号出现时,将按照三种方式中的一种处理信号:
1 忽略此信号
大多数信号都是按照这种方式进行处理的,当有两种信号却决不能被忽略。SIGKILL和SIGSTOP。这两种信号不能够被忽略的原因是:它们向超级用户提供一种终止或停止进程的方法。
2 执行用户希望的动作
通知内核在某种信号发生时,调用一个用户函数。在用户函数中,执行用户希望的处理。
3 执行系统默认动作
对大多数信号系统默认动作是终止该进程。
发送信号的主要函数有kill和raise。
区别在于:kill既可以向自身发送信号,也可以向其它进程发送信号。与kill函数不同的是,raise是向进程自身发送信号。
#include<sys/types.h>
#include<signal.h>
int kill(pid_t pid,int signo)
int raise(int signo)
Kill的pid参数有四种不同情况:
1 pid>0 :将信号发送给进程ID为pid的进程。
2 pid==0 :将信号发送给同组的进程。
3 pid<0 :将信号发送给其它进程组ID为pid绝对值的进程。
4 pid==-1 :将信号发送给所有进程。
使用alarm函数可以设置一个时间值(闹钟时间),当所设置的时间到了时,产生SIGALRM信号(发送给进程本身)。如果不捕捉此信号,则默认动作是终止改进程。
#include<unistd.h>
unsigned int alarm(unsigned int seconds)
seconds:经过指定的seconds秒后会产生信号SIGALRM。
每个进程只能有一个闹钟时间。如果在调用alarm时,之前已经为该进程设置过闹钟时间,而且他还没有超时,以前登记的闹钟时间则被新的闹钟时间替代。
如果有以前登记尚未超时的闹钟时间,而这次seconds值为0,则表示取消以前的闹钟。
pause函数是调用进程挂起知道捕捉到一个信号。
#include<unistd.h>
int pause(void)
只有执行一个信号处理函数后,挂起才结束。
当信号捕捉到某个信号时,可以忽略该信号或是使用指定的函数来处理该信号,或者使用系统默认方式。
信号处理的主要方法有两种,一种是使用简单的signal函数,另一种是使用信号集函数组。
signal函数
#include<signal.h>
void ( *signal(int signo,void(*func)(int)))(int)
如何理解?
typedef void(*sighandler_t)(int)
sighandler_t signal(int signum,sighandler_t handler)
Func可能的值是:
1 SIG_IGN:忽略此信号
2 SIG_DFL:按系统默认方式处理
3 信号处理函数名:使用该函数处理
信号实例:
程序运行步骤如下:
首先启动进程
[root@localhost 4]# ./mysignal
Waiting for signal SIGBUS
打开另外一个终端,要发送信号,首先要查看此程序的进程号,使用ps -aux命令查看
在新终端输入如下命令发送信号
[root@localhost 4]# kill -s SIGBUS 3687
终端1的结果如下:
[root@localhost 4]# ./mysignal
Waiting for signal SIGBUS
I have get SIGBUS
以上为程序运行的步骤与结果,蓝色表示为在终端1下,绿色表示在终端2下