1、信号的基本概念
信号是系统响应某个条件而产生的事件,进程接收到信号会执行相应的操作。
与信号有关的系统调用在“signal.h”头文件中有声明
2、修改信号的响应方式 – signal()
三种响应方式
3、发送信号 – kill()
kill() 可以向指定的进程发送指定的信号:
int kill(pid_t pid, int sig);
pid > 0 指定将信号发送个那个进程
pid == 0 信号被发送到和当前进程在同一个进程组的进程
pid == -1 将信号发送给系统上有权限发送的所有的进程
pid < -1 将信号发送给进程组 id 等于 pid 绝对值,并且有权限发送的所有的进程。
sig 指定发送信号的类型。
#define SIGKILL 9 //该信号的响应方式不允许改变
#define SIGTERM 15 //系统 kill 命令默认发送的信号
#define SIGCHLD 17 //子进程结束后,会默认给父进程发送该信号
perror()打印错误的原因
代码示例第一次按ctrl+c 打印sig值第二次按默认结束
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<signal.h>
void fun_sig(int sig)
{
printf("sig=%d\n",sig);
signal(SIGINT,SIG_DFL);//也可signal(sig,SIG_IGN);
}
int main(int argc,char*argv[],char* envp)
{
signal(SIGINT,fun_sig);//(有多个按照最后一个执行)
//signal(SIGINT,SIG_IGN);//忽略这一信号
while(1)
{
printf ("hello\n");
sleep(1);
}
}
kill使用示例(找到进程pid ps -ef | grep 进程名)
#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);
}
#define SIGCHLD 17 //子进程结束后,会默认给父进程发送该信号
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<signal.h>
#include<assert.h>
#include<sys/wait.h>
void fun_sig(int sig)
{
printf("sig=%d\n",sig);
int val=0;
wait(&val);//子进程先结束会产生僵死进程wait处理僵死进程 如果有多个子进程还可优化
}
int main(int argc,char*argv[],char* envp)
{
signal(SIGCHLD,fun_sig);
//signal(SIGCHLD,SIG_IGN);也可忽略子进程从而达到处理僵死进程需屏蔽上一行
pid_t pid =fork();
assert(pid !=-1);
if(pid==0)
{
sleep(3);
exit(0);
}
for(int i=0;i<5;++i)
{
printf ("main runn\n");
sleep(1);
}
}