Linux信号的使用

本文详细介绍了Linux信号的使用,包括信号的概念、分类、产生、响应方式和发送。重点讨论了信号的分类,如可靠信号与不可靠信号,以及如何修改信号响应方式。此外,还阐述了信号的发送,如通过键盘组合键、硬件异常和系统调用产生信号。最后,通过实例展示了如何利用信号异步处理僵尸进程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1.1概念

信号:操作系统预先定义好的某些特定的事件,信号可以被产生,也可以被接收。产生和接收的主体都是进程。(一个进程向另一个进程通知某一个事件的发生)。

kill -l//查看所有信号

在这里插入图片描述
1~34为普通信号,不存在(0,32,33号信号),34以上为实时信号
Linux系统平台上信号的定义:
每个信号都有一个编号和一个宏定义名称,这些宏定义可以在系统调用signal.h找到。
在这里插入图片描述
在这里插入图片描述

常见信号的值及对应功能。
在这里插入图片描述

1.2信号分类

将信号分为可靠信号(实时信号)和不可靠信号(非实时信号):
可靠信号:Linux改进了信号机制,新增了32种信号,均为可靠信号,信号支持排队,不会丢失,发多少次,就可以收到多少次,信号值位[SIGRTMIN,SIGRTMAX]区间都是可靠信号。
非可靠信号:从unix系统继承过来的信号都是非可靠信号,信号不支持排队,可能会丢失,比如发送多次相同的信号,进程只能收到一次,信号值小于SIGRTMIN的都是非可靠信号。

1.2信号的产生

1、通过键盘组合键产生,比如ctrl+c产生,SIGINT信号,ctrl+\产生SIGQUIT信号,ctrl+z产生SIGTSTP信号。
2、硬件异常产生信号,这些条件由硬件检测,并通知内核,然后内核向进程发送适当的信号,如执行除以0指令,进程访问了非法的内存,cpu运算单元都会产生异常,内核将这个异常解释成一个个信号发送给进程。
3、一个进程调用kill(2)函数可以发送信号给另一个进程。

1.3信号的响应方式

进程收到信号后,可以选择的处理动作有以下三种:
1、忽略此信号
2、执行该信号的默认处理动作
3、提供一个信号处理函数(自定义),要求内核在处理该信号时切换到用户态执行这个处理函数,这种方式称为捕捉一个信号。
4、大多数信号都可使用这种方式进行处理,但是SIGKILL和SIGSTOP不能被忽略,因为他们向超级用户提供了使进程终止或停止的可靠方法。
在这里插入图片描述

修改信号的响应方式

修改系统响应方式的系统调用

#include<signal.h>
typedef void (*sighandler_t)(int);
sighandler_t signal(int signum,sighandler_t handler);

signum : 信号类型,也是信号对应的宏值
handler: 函数指针,可以是SIG_IGN,内核忽略此信号;如果是SIG_DFL,则表示此信号按照系统默认处理方式处理;可以是用户自定义函数,表示在信号发生时,调用该函数,该函数的格式必须是:一个整型参数,无返回值。

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<assert.h>
#include<signal.h>

void fun(int sig)
{
	printf("sig=%d\n",sig);
}


int main(int argc,char *argv[],char* envp[])
{
	signal(SIGINT,fun);
	while(1)
	{
		printf("hello\n");
		sleep(1);
	}
	exit(0);
}

运行此代码,按下ctrl+c时,本来会产生信号,终端中断,但是当前进程输出sig = 2
在这里插入图片描述
进程第一次接收ctrl+c发送信号打印sig第二次收到信号结束进程。

void fun(int sig)
{

	printf("sig=%d\n",sig);
	signal(SIGINT,SIG_DFL);
}


int main()
{
	signal(SIGINT,fun);
	while(1)
	{
		printf("hello\n");
		sleep(1);
	}
	exit(0);
}

当第一次收到信号是我们执行的fun方法的时候,第二次收到信号其实就是在fun方法执行过程中。

在这里插入图片描述

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<assert.h>
#include<signal.h>

void fun(int sig)
{

	printf("start\n");
	sleep(5);
	printf("end\n");
}


int main()
{
	signal(SIGINT,fun);
	while(1)
	{
		printf("hello\n");
		sleep(1);
	}
	exit(0);
}

在信号函数的执行过程中,该信号如果多次触发,系统只能记录一次。信号处理函数执行过程中,该信号会被屏蔽。
在这里插入图片描述

1.4信号的发送

系统调用:

int kill(pid_t pid, int signum);

raise函数则允许进程向自身发送信号:

int raise(int signo);

pid:将信号发送给那个进程
signum:发送信号的类型
成功返回0,失败返回-1;
pid > 0 指定接收信号进程的PID

pid == 0 将信号发送给当前进程组中的所有进程

pid == -1 将信号发送给系统上所有的进程(有权限发送)

pid < -1将信号发送给进程组ID为-pid

kill pid //杀死进程
kill -9 pid //强制杀死进程

通过kill函数发送信号实现类似终端kill的相关命令
1、kill系统默认调用SIGTERM(15号)终止指定进程2、加参数-9 系统调用SIGKILL(9)

void fun(int sig)
{

	printf("sig=%d\n",sig);
	signal(SIGINT,SIG_DFL);
}


int main()
{
	signal(SIGINT,fun);
	while(1)
	{
		printf("hello\n");
		sleep(1);
	}
	exit(0);
}
include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<assert.h>
#include<signal.h>


int main(int argc,char *argv[],char *envp[])
{
	if(argc != 3)
	{
		printf("main 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);
}

在这里插入图片描述
用mykill终止进程
在这里插入图片描述

由图可以看到这个进程使用kill,并没有杀死,要用kill-9
在这里插入图片描述
在这里插入图片描述
用我们自己模拟实现的:
在这里插入图片描述

1.5利用信号异步处理僵尸进程

#define SIGCHID 17//子进程结束后默认给父进程发送信号
void fun(int sig)
{
	printf("sig= %d\n",sig);
	wait(NULL);
}



int main(int argc,char *argv[],char *envp[])
{
	char *s = NULL;
	int n = 0;
	pid_t pid = fork();
	signal(SIGCHLD,fun);
	assert(pid != -1);
	if(pid == 0)
	{
		s = "child";
		n = 1;
	}
	else
	{
		s = "parents";
		n = 7;
	}

	int i = 0;
	for( ;i < n;i++)
	{
		printf("s=%s\n",s);
		sleep(1);
	}
	exit(0);
}

产生僵尸进程
在这里插入图片描述
父进程要处理僵尸进程,必须调用wait方法。父进程在子进程结束后调用wait方法,父进程不知道子进程什么时候结束,因此子进程结束后,给父进程发送一个信号,父进程可以在信号处理函数中调用wait方法。
在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值