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。

完整示例

signal函数的定义很复杂,但是它的用法还是比较简单的,你记住它的两个参数就可以了。signal有两个参数sig和func,signal这个函数是用来接收信号并处理的,所以sig参数表示将要处理哪种类型的信号,而func参数是一个函数指针,用来指定信号的处理函数,也就是当程序接收到sig那个类型的信号后,就会调用func指针指向的函数。func指针的原型是:

void (*func) (int)

所以信号的处理函数必须是一个返回void,只有一个int类型参数的函数。

比如如果程序需要处理Ctrl+C组合键产生的信号,就可以这样使用signal函数:

(void) signal(SIGINT, myfunc)

而myfunc函数可以这样定义:

void myfunc(int sig)

{

printf("Hello, the signal is %d\n", sig)

// 因为现在处理的是Ctrl+C信号,所以下面要

// 恢复程序对Ctrl+C的默认反应

(void) signal(SIGINT, SIG_DFL)

}

signal,此函数相对简单一些,给定一个信号,给出信号处理函数则可,当然,函数简单,其功能也相对简单许多,简单给出个函数例子如下:

   #include <signal.h>  

   #include <stdio.h>  

   #include <unistd.h>  

     

   void ouch(int sig)  

   {  

       printf("I got signal %d\n", sig)  

       // (void) signal(SIGINT, SIG_DFL)  

      //(void) signal(SIGINT, ouch)  

    

  }  

       

 int main()  

 {  

     (void) signal(SIGINT, ouch)  

  

     while(1)  

    {  

        printf("hello world...\n")  

         sleep(1)  

     }  

 }

当然,实际运用中,需要对不同到signal设定不同的到信号处理函数,SIG_IGN忽略/SIG_DFL默认,这俩宏也可以作为信号处理函数。同时SIGSTOP/SIGKILL这俩信号无法捕获和忽略。注意,经过实验发现,signal函数也会堵塞当前正在处理的signal,但是没有办法阻塞其它signal,比如正在处理SIG_INT,再来一个SIG_INT则会堵塞,但是来SIG_QUIT则会被其中断,如果SIG_QUIT有处理,则需要等待SIG_QUIT处理完了,SIG_INT才会接着刚才处理。


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

原文地址: https://outofmemory.cn/yw/6138594.html

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

发表评论

登录后才能评论

评论列表(0条)

保存