linux——Signal ()函数用法和总结

void(* signal(int sig,void(* func)(int)))(int);

设置处理信号的功能

指定使用sig指定的信号编号处理信号的方法。 参数func指定程序可以处理信号的三种方式之一:

  • 默认处理(SIG_DFL):信号由该特定信号的默认动作处理。
  • 忽略信号(SIG_IGN):忽略信号,即使没有意义,代码执行仍将继续。
  • 函数处理程序:定义一个特定的函数来处理信号。

或 SIG_DFL 要么 SIG_IGN 被设置为程序启动时每个支持信号的默认信号处理行为。

参数

SIG设置处理功能的信号值。以下宏常量表达式标识标准信号值:

信号
SIGABRT(信号中止)异常终止,例如由...发起 退出 功能。
SIGFPE(信号浮点异常)错误的算术运算,例如零分频或导致溢出的运算(不一定是浮点运算)。
SIGILL(信号非法指令)无效的功能图像,例如非法指令。这通常是由于代码中的损坏或尝试执行数据。
SIGINT(信号中断)交互式注意信号。通常由应用程序用户生成。
SIGSEGV(信号分段违规)对存储的无效访问:当程序试图在已分配的内存之外读取或写入时。
SIGTERM(信号终止)发送到程序的终止请求。

每个库实现可以提供可以与此函数一起使用的附加信号值宏常量。

请注意,并非所有运行环境都需要生成自动信号,即使在上述特定情况下也是如此,尽管所有运行环境都必须通过显式调用生成的信号来生成提高 功能。FUNC指向函数的指针。这可以是程序员定义的函数,也可以是以下预定义函数之一:

SIG_DFL默认处理:信号由该特定信号的默认操作处理。
SIG_IGN忽略信号:忽略信号。

如果是一个函数,它应该遵循以下原型(使用C链接):

void handler_function (int parameter);

返回值

返回类型与参数func的类型相同。

如果请求成功,则该函数返回指向特定处理函数的指针,该函数在调用之前负责处理该信号(如果有的话)。或者SIG_DFL 要么 SIG_IGN如果在调用之前信号由默认处理程序处理或被忽略,则相应地。

如果该功能未能成功注册新的信号处理程序,则返回SIG_ERR 和 错误号 可以设置为正值。

【文章福利】小编推荐自己的Linux内核技术交流群: 【977878001】整理一些个人觉得比较好得学习书籍、视频资料!进群私聊管理领取 内核资料包(含视频教程、电子书、实战项目及代码)

内核资料直通车:Linux内核源码技术学习路线+视频教程代码资料

免费加入学习:Linux/c/c++/内核源码/音视频/DPDK/Golang云原生/QT

/* signal example */
#include <stdio.h>      /* printf */
#include <signal.h>     /* signal, raise, sig_atomic_t */
 
sig_atomic_t signaled = 0;
 
void my_handler (int param)
{
  signaled = 1;
}
 
int main ()
{
  void (*prev_handler)(int);
 
  prev_handler = signal (SIGINT, my_handler);
 
  /* ... */
  raise(SIGINT);
  /* ... */
   
  printf ("signaled is %d.\n",signaled);
   
 
  return 0;
}

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

第一个参数是要捕捉的信号(查看信号:kill -l,9号SIGKILL信号不能被捕捉);

第二个参数表示我们要对信号进行的处理方式。

信号的处理方式一般有三种:

(1)忽略此信号(SIG_IGN):

#include<stdio.h>
#include<signal.h>
int main()
{
    signal(2,SIG_IGN);
    while(1)
    {
        printf("2333\n");
        sleep(1);
    }
    return 0;
}

当执行程序时,陷入死循环,此时按下Ctrl+c进程并不会停止,因为我们对Ctrl+c产生的2号SIGINT信号采取了忽略处理,若要停止进程可用Ctrl+z(SIGQUIT);

运行结果如下:

(2)执行该信号的默认处理动作(SIG_DFL):

#include<stdio.h>
#include<signal.h>
int main()
{
    signal(2,SIG_DFL);
    while(1)
    {
        printf("2333\n");
        sleep(1);
    }
    return 0;
}

当执行程序时,同样是死循环,此时按下Ctrl+z进程停止,因为我们对2号信号采取默认动作处理,系统默认2号信号终止进程。

(3提供一个信号处理函数,要求内核在处理该信号时切换到用户态执行这个处理函数,这种方式称为捕捉(catch)一个信号:

#include<stdio.h>
#include<signal.h>
void  handler(int signo)//自定义一个函数处理信号
{
    printf("catch a signal:%d\n:",signo);
}
int main()
{
    signal(2,handler);
    while(1)
    {
        printf("1111\n");
        sleep(1);
    }
    return 0;
}

运行结果如图:

一些常用的Signal :

SignalDescription
SIGABRT由调用abort函数产生,进程非正常退出
SIGALRM用alarm函数设置的timer超时或setitimer函数设置的interval timer超时
SIGBUS某种特定的硬件异常,通常由内存访问引起
SIGCANCEL由Solaris Thread Library内部使用,通常不会使用
SIGCHLD进程Terminate或Stop的时候,SIGCHLD会发送给它的父进程。缺省情况下该Signal会被忽略
SIGCONT当被stop的进程恢复运行的时候,自动发送
SIGEMT和实现相关的硬件异常
SIGFPE数学相关的异常,如被0除,浮点溢出,等等
SIGFREEZESolaris专用,Hiberate或者Suspended时候发送
SIGHUP发送给具有Terminal的Controlling Process,当terminal 被disconnect时候发送
SIGILL非法指令异常
SIGINFOBSD signal。由Status Key产生,通常是CTRL+T。发送给所有Foreground Group的进程
SIGINT由Interrupt Key产生,通常是CTRL+C或者DELETE。发送给所有ForeGround Group的进程
SIGIO异步IO事件
SIGIOT实现相关的硬件异常,一般对应SIGABRT
SIGKILL无法处理和忽略。中止某个进程
SIGLWP由Solaris Thread Libray内部使用
SIGPIPE在reader中止之后写Pipe的时候发送
SIGPOLL当某个事件发送给Pollable Device的时候发送
SIGPROFSetitimer指定的Profiling Interval Timer所产生
SIGPWR和系统相关。和UPS相关。
SIGQUIT输入Quit Key的时候(CTRL+\)发送给所有Foreground Group的进程
SIGSEGV非法内存访问
SIGSTKFLTLinux专用,数学协处理器的栈异常
SIGSTOP中止进程。无法处理和忽略。
SIGSYS非法系统调用
SIGTERM请求中止进程,kill命令缺省发送
SIGTHAWSolaris专用,从Suspend恢复时候发送
SIGTRAP实现相关的硬件异常。一般是调试异常
SIGTSTPSuspend Key,一般是Ctrl+Z。发送给所有Foreground Group的进程
SIGTTIN当Background Group的进程尝试读取Terminal的时候发送
SIGTTOU当Background Group的进程尝试写Terminal的时候发送
SIGURG当out-of-band data接收的时候可能发送
SIGUSR1用户自定义signal 1
SIGUSR2用户自定义signal 2
SIGVTALRMsetitimer函数设置的Virtual Interval Timer超时的时候
SIGWAITINGSolaris Thread Library内部实现专用
SIGWINCH当Terminal的窗口大小改变的时候,发送给Foreground Group的所有进程
SIGXCPU当CPU时间限制超时的时候
SIGXFSZ进程超过文件大小限制
SIGXRESSolaris专用,进程超过资源限制的时候发

原文作者:cs_wu

原文地址:Signal ()函数用法和总结(版权归原文作者所有,侵权留言联系删除)

  • 0
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值