linux系统上信号发送和信号接收讲解

linux系统上信号发送和信号接收讲解,第1张

用于进程间通信,通信机制由 *** 作系统保证,比较稳定。

在linux中可以通过kill -l查看所有信号的类型。

kill -信号类型 进程ID

int kill(pid_t pid, int sig)

入参pid :

pid >0: 发送信号给指定的进程。

pid = 0: 发送信号给 与调用kill函数进程属于同一进程组的所有进程。

pid <0: 取|pid|发给对应进程组。

pid = -1:发送给进程有权限发送的系统中所有进程。

sig :信号类型。

返回值 :成功:0;失败:-1 (ID非法,信号非法,普通用户杀init进程等权级问题),设置errno

以OpenHarmony源码为例,应用ANR后,AbilityManagerService会通知应用dump堆栈信息,就是通过信号量做的。

头文件位置 :

include <signal.h>

函数解释 :

typedef void (*sighandler_t)(int)

sighandler_t signal(int signum, sighandler_t handler)

当接收到指定的信号signum时,就会跳转到参数handler指定的函数执行。其中handler的入参是信号值。

函数原型

signum参数指出要捕获的信号类型,act参数指定新的信号处理方式,oldact参数输出先前信号的处理方式(如果不为NULL的话)。

sigaction结构体

sa_handler 信号处理函数

sa_mask 在处理该信号时可以暂时将sa_mask 指定的信号集搁置

sa_flags 指定一组修改信号行为的标志。 它由以下零个或多个的按位或组成

   SA_RESETHAND:当调用信号处理函数时,将信号的处理函数重置为缺省值SIG_DFL

   SA_RESTART:如果信号中断了进程的某个系统调用,则系统自动启动该系统调用

   SA_NODEFER :一般情况下, 当信号处理函数运行时,内核将阻塞该给定信号。但是如果设置了 SA_NODEFER标记, 那么在该信号处理函数运行时,内核将不会阻塞该信号

sa_restorer 是一个替代的信号处理程序,当设置SA_SIGINFO时才会用它。

相关函数

int sigemptyset( sigset_t *set)

sigemptyset()用来将参数set信号集初始化并清空。

执行成功则返回0,如果有错误则返回-1。

完整示例

纠正一下:

输出in sig_fun1:30就是第二次调用的时候输出的,第一次调用只是绑定了SIGUSR1的信号处理函数,不会进入该处理函数

为什么会有这样的输出呢?

signal函数是将信号与处理函数进行绑定,成功绑定则返回绑定之前的信号处理函数。那么来看看你的代码,第一次调用将sig_fun1绑定,无输出;第二次调用将sig_fun2绑定,也就是把sig_fun1替换下来,并且你还调用了它,参数为30,所以会有那样的输出。

该如何改呢?

其实你并没有涉及到linux的信号处理机制,光绑定是不够的,还需要发信号给它,才能真正进入信号处理过程。给你一个示例代码吧

#include<signal.h>

#include<stdlib.h>

#include<stdio.h>

#include<unistd.h>

void sig_fun2(int signo)

{

printf("in sig_fun2:%d\n", signo)

}

void sig_fun1(int signo)

{

printf("in sig_fun1:%d\n", signo)

}

int main()

{

unsigned long i

if (signal(SIGUSR1, sig_fun1) == SIG_ERR)

{

printf("signal fun1 error\n")

exit(1)

}

sleep(15)

(signal(SIGUSR1, sig_fun2))(30)

sleep(15)

printf("done\n")

return 0

}

/****************************C 代码完,下面是如何运行***************************/

首先编译,假设生成可执行程序为test

然后运行,我用的是后台运行: nohup ./test>output.txt &

注意,这种方法要将输出重定向到文件output.txt(名字无所谓),然后你会看到一个数字,就是pid进程号

最后,在15秒之内发送信号:kill -SIGUSR1 进程号

现在你就可以打开output.txt看输出结果了。如果用sleep的话会被打断,所以只有两个输出加上替换处理函数时的输出共3个,也可以换成 int n=15while(n--)sleep(1)

-------------------------------------------------------------

怎么样,加分吧

-------------------------------------------------------------

1.我就是想问第二次绑定sig_fun2的时候,调用了第一次绑定的sig_fun1么?

调用了, (signal(SIGUSR1, sig_fun2))(30)就是这一句, signal(SIGUSR1, sig_fun2)是个函数指针,你这样写就是调用它了,但是这和信号处理没关系,写成signal(SIGUSR1, sig_fun2)就可以了

这就是你所说的成功则返回绑定之前的函数???

那当时绑定sig_fun1的时候,返回之前的处理函数是什么??

这个就是系统默认的了,比如SIGINT就是你ctrl+c取消程序执行发送的信号,它的处理函数就是结束程序的一系列动作,不过SIGUSR1是留给用户自定义的信号,系统默认应该是啥也不做的一个函数,例如void fun(int signo){},你也可以第一次绑定的时候就调用试试看对不对

2.还有我在看signal函数定义的时候,void(//...)(int) 最后传入的这个int整形参数就是我们自定义sig_fun()中所接收的30么??我看例子里面有的signal(SIGINT,myfunc);也没有带参数啊,搞不懂

是你理解错了,signal函数只是绑定,没涉及到调用绑定函数,不用带参数,信号处理函数不是像你这样调用的。callback回调你知道吧,就是先做好一个函数或过程放着,事件触发的时候才调用。那个30是你用普通函数调用的方式时的参数,跟信号处理一点关系也没有,你用60,70也没半毛钱关系。我猜你是想要调用信号处理函数,然后迷糊了,其实我上面说的“kill -SIGUSR1 进程号”就是触发程序调用该处理函数的信号,这和kill -9 杀死进程一个道理,只不过处理函数不同,结果不一样。ctrl+c也可以用信号的方式发送,kill -2 进程号,或者 kill -SIGINT 进程号


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/yw/8574555.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-04-18
下一篇 2023-04-18

发表评论

登录后才能评论

评论列表(0条)

保存