Linux C进程与多线程入门—(3)信号
概念
信号是系统响应某些状况而产生的事件,进程在接受到信号时会采取相应的行动。信号可以明确地由一个进程产生发送到另外一个进程,用这种办法传递信息或协调操作行为。
进程可以生成信号、捕捉并相应信号或屏蔽信号。信号的名称是在头文件signal.h里定义。下面我列出一部分,如下:
———————————————————————————
信号名称 说明
———————————————————————————
SIGALRM 警告钟
SIGHUP 系统挂断
SIGINT 终端中断
SIGKILL 停止进程(此信号不能被忽略或捕获)
SIGPIPE 向没有读者的管道写数据
SIGQUIT 终端退出
SIGTERM 终止
———————————————————————————
如果进程接收到上表中的某个信号但实现并没有安排捕捉它,进程就会立刻终止。
函数
#include <signal.h> void (*signal(int sig, void (*func) (int) ))(int);
这个函数看起来很复杂,但使用却是很简单的。第一个参数sig就是准备捕获或屏蔽的信号,接收到指定信号时将调用func函数处理。。
实例1—处理SIGINT信号
ctrlc.c文件
#include <signal.h> #include <stdio.h> #include <unistd.h> void ouch(int sig) { /*此处,signal(SIGINT, SIG_DFL),SIG_DFL表示 ouch函数捕获到SIGINT信号,作出输出信息处理之后,恢复了SIGINT的默认行为*/ printf("OUCH! - I got signal %d\n", sig); (void) signal(SIGINT, SIG_DFL); } int main(){ (void) signal(SIGINT, ouch); while(1) { printf("Hello World!\n"); sleep(1); } }
这个程序就是截获组合键Ctrl+C产生的SIGINT信号。没有信号出现时,它每隔一秒就会输出一个消息。第一次按下Ctrl+C产生的SIGINT信号,
程序会调用ouch函数,输出信息,同时,恢复SIGINT为默认行为(即按下Ctrl+C组合键后即结束运行),那么第二次按下Ctrl+C组合键时,程序就结束了运行。
运行效果见下图:

实例2—模仿闹钟行为
函数1
#include <sys/types.h> #include <signal.h> int kill (pid_t pid, int sig);
kill函数的作用是把sig信号发送给标识为pid的进程,成功时返回“0”,失败时返回“-1”. 要想发送一个信号,两个进程(发送和接受两方)必须拥有同样的用户ID,,就是说,
你只能想自己的另一个进程发送信号。但是超级用户可以向任何进程发送信号。
函数2
#include <unistd.h> unsigned int alarm (unsigned int seconds ) ;
alarm函数是在seconds秒后安排发送一个SIGALARM信号。若seconds为0,表示将取消全部已经设置的闹钟请求。每一个进程只有一个可用的闹钟。
它的返回值是前一个闹钟闹响之前还需经过的剩余秒数。调用失败则返回“-1”.
#include <signal.h> #include <stdio.h> #include <unistd.h> static int alarm_fired = 0; void ding(int sig) { alarm_fired = 1; } int main() { int pid; printf("闹钟程序已经启动\n"); /*子进程休眠5秒后向父进程发送SIGALARM信号,然后结束进程*/ if((pid = fork()) == 0) { sleep(5); kill(getppid(), SIGALRM); exit(0); } /*父进程执行的内容*/ printf("5秒后闹铃启动\n"); (void) signal(SIGALRM, ding); /*将运行的程序挂起,直到接收到信号为止*/ pause(); if (alarm_fired) printf("Ding!\n"); printf("done\n"); exit(0); }
程序通过fork启动一个新进程,这个紫禁城休眠5秒后向 自己的父进程发送一个SIGALARM信号。父进程在安排好捕捉SIGALARM信号后暂停运行,直到接收到一个信号为止。
运行结果见下图:

参考文献:Linux程序设计