信号的基本概念
在一个进程中可能会遇到各种各样的事件,比如:终端中断,终止等等。
信号就是用来通知进程产生了某个事件。
那么我们如何使用信号呢?
首先,我们要通过系统调用kill()发送信号,然后我们在通过signal()相应信号。
响应信号有三种方式:1.默认 2.忽略 3.自定义
与信号有关的系统调用在“signal.h”头文件中有声明。
常见信号的值,及对应的功能说明:
信号的值在系统源码中的定义如下:
#define SIGHUP 1
#define SIGINT 2 //键盘按下 Ctrl+c 时,会产生该信号
#define SIGQUIT 3
#define SIGILL 4
#define SIGTRAP 5
#define SIGABRT 6
#define SIGIOT 6
#define SIGBUS 7
#define SIGFPE 8
#define SIGKILL 9 //该信号的响应方式不允许改变
#define SIGUSR1 10
#define SIGSEGV 11
#define SIGUSR2 12
#define SIGPIPE 13 //读端关闭的描述符,写端写入时产生,该信号会终止程序
#define SIGALRM 14
#define SIGTERM 15 //系统 kill 命令默认发送的信号
#define SIGSTKFLT 16
#define SIGCHLD 17 //子进程结束后,会默认给父进程发送该信号
#define SIGCONT 18
#define SIGSTOP 19
#define SIGTSTP 20
#define SIGTTIN 21
#define SIGTTOU 22
#define SIGURG 23
修改信号的响应方式– signal()
信号响应函数signal(),有两个参数。
第一个参数是信号的代号sig,第二个参数是参数为int返回值为void的信号处理函数。
如果是默认的响应方式,那么就在第二个参数写成SIG_DFL。
如果是忽略的响应方式,那么就在第二个参数写成SIG_IGN。
如果是自定义的响应方式,那么就在第二个参数写成自己写的信号处理函数。
在键盘上按下 Ctrl+c 时,会给当前终端前台执行的进程发送 SIGINT 信号,用 signal 修 改 SIGINT 信号的响应方式示例代码如下:
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<assert.h>
#include <signal.h>
void fun(int sign)
{
printf("fun was called, sign = %d\n", sign);
}
int main()
{
signal(SIGINT, fun);
while(1)
{
sleep(1);
printf("main running\n");
}
exit(0);
}
运行结果如下:
可见,无论我们怎么用Ctrl+c ,都不会再中断,而是会按照我们所写的fun函数打印fun was called,且信号值为2。
发送信号 – kill()
kill() 可以向指定的进程发送指定的信号:
int kill(pid_t pid, int sig);
第一个参数是进程id号pid:
pid > 0 指定将信号发送个那个进程
pid == 0 信号被发送到和当前进程在同一个进程组的进程
pid == -1 将信号发送给系统上有权限发送的所有的进程
pid < -1 将信号发送给进程组 id 等于 pid 绝对值,并且有权限发送的所有的进程。
sig 指定发送信号的类型。
使用 kill()系统调用实现类似于系统 kill 命令的程序如下:
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <assert.h>
#include <signal.h>
int main(int argc, char* argv[])//pid, sig
{
if ( argc != 3 )
{
printf("argc error\n");
exit(0);
}
int pid = 0;
int sig = 0;
sscanf(argv[1],"%d",&pid);
sscanf(argv[2],"%d",&sig);
if ( kill(pid,sig) == -1 )
{
perror("kill error");
}
exit(0);
}
通过这个kill()函数给上面自定义的signal()函数发送信号,运行结果如下: