1、信号通信
信号(signal)机制是Unix系统最为古老的进程间通信机制,很多条件可以产生信号:
(1)单用户按某些按键时,产生信号。
(2)硬件异常产生信号:除数为0、无效的存储访问等等。这些情况由硬件检测到,将其通知内核,然后内核产生适当的信号通知进程,例如,内核对正在访问一个无效存储区的进程产生一个SIGEGV信号。
(3)进程用 kill 函数将信号发送给另一个进程。
(4)用户可以用kill命令将信号发送给其他进程。
2、信号类型
几种常见的信号(signum):
SIGINT : 来自键盘(Ctrl + c)的中断信号。
SIGKILL:该信号结束接收信号的进程。
SIGHUP:从终端发出的结束信号。
SIGTERM:kill 命令发出的信号。
SIGCHLD:标志子进程停止或结束的信号。
SIGSTOP:来自键盘(Ctrl + z)或调试程序的停止执行信号。
3、信号处理
当某信号出现时,将按照以下三种方式中的一种进行处理:
(1)忽略此信号:大多数信号都按照这种方式进行处理,但是有两种信号决不能被忽略– SIGKILL 和 SIGSTOP。因为它们向 root 用户提供了一种终止进程的方法。
(2)执行用户期望的动作:通知内核在某种信号发生时,调用一个用户函数,在用户函数中,执行用户期望的处理。
(3)执行系统默认的动作:绝大多数信号的系统默认动作是终止该进程。
#include <signal.h>
void (*signal(int signum, void(*func)(int)))(int);
其中:
typedef void (*sighandler_t)(int)
sighandler_t (int signum, sighandler_t handler)
func可能的值是:
1、SIG_IGN:忽略此信号。
2、SIG_DEL:安系统默认方式处理。
3、信号处理函数:使用用户处理函数处理。
4、信号发送
发送信号的主要函数有 kill 和 raise。
二者区别:
kill 既可以向自身发送信号,也可以向其他进程发送信号。与 kill 函数不同的是,raise函数只向自身发送信号。
#include <sys/types.h>
#include <signal.h>
int kill(pid_t pid, int signum);
int raise(int signum);
kill 函数 pid 参数四种不同的情况:
(1)pid > 0
将信号发送给进程 ID 为 pid 的进程。
(2)pid == 0
将信号发送给同组的进程。
(3)pid < 0
将进程发送给进程 ID 等于 pid 绝对值的进程。
(4)pid == -1
将进程发送给所有进程。
#include <unistd.h>
unsigned int alarm(unsigned int seconds);
功能:经过 seconds 秒后会产生一个SIGALARM信号。
#include <unistd.h>
int pause(void);
功能:pause函数是调用进程挂起直至捕捉一个信号才结束进程。
5、demo
#include <apue.h>
void handlerFunc(int signum)
{
if (signum == SIGINT){
printf("I get a SIGINT\n");
}
else if (signum == SIGQUIT){
printf("I get a SIGQUIT\n");
}
}
int main(int argc, const char *argv[])
{
printf("Waiting for signal SIGINT or SIGQUIT\n");
/* 注册信号处理函数 */
signal(SIGINT, handlerFunc);
signal(SIGQUIT, handlerFunc);
/* 等待信号 */
pause();
return 0;
}
在终端输入ps -aux
即可查找进程号,然后在终端输入kill -s SIGINT [pid]
或者 kill -s SIGQUIT [pid]
即可产生一个 SIGINT 信号或者 SIGQUIT 信号。