1 信号
信号是UNIX和Linux系统响应某些条件而产生的一个事件,接收到信号的进程会采取一些行动。信号是由于某些错误条件而生成的,如内存段冲突、浮点处理器错误或非法指定等。信号由shell或终端处理器生成来引起中断,如果进程接收到信号而没有安排捕获它,进程会立刻终止。通常系统将在当前目录下生成核心转储文件core,该文件是进程在内存中的映像,有助于程序调试。
信号名称在头文件/usr/include/signal.h
中定义,以SIG开头。
kill
命令可向非前台进程发送信号,需要有信号名称和目标进程的PID。
killall
命令可以给运行着某一命令的所有进程发送信号,当不知道某个进程的PID,或是想给执行相同命令的许多不同进程发送信号,可以使用该命令。
#include <signal.h>
void(*signal(int sig,void(*func)(int)))(int);
signal函数用来对信号进行处理。sig参数指定想要捕获的信号,func函数指针指向接收到指定的信号后将要调用的函数,函数*func的参数为接收到的信号代码(SIGINT为2,如果想要在一个函数中处理多个信号,该参数是很有用的)。返回一个同类型的函数,即先前用来处理这个信号的函数指针,如果未定义信号处理函数则返回SIG_ERR并设置errno,如果给出的是一个无效信号,或者尝试处理的信号是不可捕获或不可忽略的信号(如SIGKILL),errno将被设置为EINVAL。func可用两个宏代替:SIG_IGN:忽略该信号,SIG_DFL:恢复默认行为。
#include <signal.h>
#include <stdio.h>
#include <unistd.h>
void ouch(int sig)
{
printf("OUCH!!!! I got signal %d\n",sig);
signal(SIGINT,SIG_DFL);
}
int main()
{
signal(SIGINT,ouch);
while(1)
{
printf("Hello world\n");
sleep(1);
}
}
在信号处理函数中使用printf是不安全的,我们可以在信号处理函数中设置一个标志,在主程序中检查该标志,如需要则在主程序中打印一条信息。
#include <signal.h>
#include <stdio.h>
#include <unistd.h>
static int flag = 0;
void ouch(int sig)
{
flag = 1;
signal(SIGINT,SIG_DFL);
}
int main()
{
signal(SIGINT,ouch);