signal函数:
函数作用:注册信号捕捉函数
函数原型:
typedef void (*sighandler_t)(int);
sighandler_t signal(int signum, sighandler_t handler);
函数参数:
- signum:信号编号
- handler:信号处理函数
测试:给没有读端的管道写数据,会产生一个SIGPIPE信号
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<sys/types.h>
#include<unistd.h>
#include <signal.h>
//信号处理函数
void handler(int signo)
{
printf("signo=[%d]\n",signo);
}
int main()
{
int fd[2];
//创建管道
int ret=pipe(fd);
if(ret<0)
{
perror("pipe error");
return -1;
}
// typedef void (*sighandler_t)(int);
// sighandler_t signal(int signum, sighandler_t handler);
//给内核区注册SIGPIPE信号处理函数
signal(SIGPIPE,handler);
//关闭读端
close(fd[0]);
write(fd[1],"hhh",strlen("hhh"));
close(fd[1]);
return 0;
}
结果:
我们通过kill -l命令查看一下SIGPIPE信号的编号
所以我们可知往关闭读端的管道写数据时会返回SIGPIPE信号。
kill函数/命令
作用:给指定进程发送指定信号
kill命令:kill -SIGKILL 进程PID
kill函数原型: int kill(pid_t pid, int sig);
返回值:
- 成功返回0
- 失败返回-1
函数参数:
1 sig:信号编号
2 pid:
- pid>0:给指定的进程发送信号
- pid=0:发送信号给与调用kill函数进程(当前进程)属于同一进程组的所有进程
- pid<0;取|pid|发给对应的进程组
- pid=-1:发给进程有权限发送的系统中所有进程
进程组:每个进程都属于一个进程组,进程组是一个或多个进程集合,他们相互关联,共同完成一个实体任务,每个进程组都有一个进程组组长(父进程),默认进程组ID与进程组组长ID相同 。
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<sys/types.h>
#include<unistd.h>
#include <signal.h>
int main()
{
pid_t pid=fork();
if(pid<0)
{
perror("fork error");
return -1;
}
else if(pid>0)
{
printf("father:pid=[%d]\n",getpid());
//父进程发送信号杀死子进程
// kill(pid,SIGKILL);
sleep(10);
}
else if(pid==0)
{
printf("child:pid=[%d],fpid=[%d]\n",getpid(),getppid());
//子进程发送信号杀死父进程
kill(getppid(),SIGKILL);
sleep(10);
}
return 0;
}
父进程杀死子进程的结果:
:
子进程杀死父进程的结果:
ps -aux命令也发现只有子进程的sleep(10)还存在,父进程已经被杀死,子进程被领养。
raise函数:
作用:给当前进程发送指定信号(自己给自己发)
函数原型: int raise(int sig);
函数返回值:成功返回0,失败返回非0值
raise(sig)= kill(getpid(), sig);
abort函数:
函数作用:给自己发送异常终止信号 6)SIGABRT,并产生core文件
函数原型: void abort(void);
abort=kill(getpid(),SIGABRT)