signal

函数名: signal  表头文件 #include<signal.h>

  功 能: 设置某一信号的对应动作

  函数原型 :void (*signal(int signum,void(* handler)(int)))(int);

  或者:typedef void(*sig_t) ( int );

  sig_t signal(int signum,sig_t handler);

  参数说明

  第一个参数signum指明了所要处理的信号类型,它可以取除了SIGKILL和SIGSTOP外的任何一种信号。

  第二个参数handler描述了与信号关联的动作,它可以取以下三种值:

  (1)一个返回值为正数的函数地址

  此函数必须在signal()被调用前申明,handler中为这个函数的名字。当接收到一个类型为sig的信号时,就执行handler 所指定的函数。这个函数应有如下形式的定义:

  intfunc(int sig);

  sig是传递给它的唯一参数。执行了signal()调用后,进程只要接收到类型为sig的信号,不管其正在执行程序的哪一部分,就立即执行func()函数。当func()函数执行结束后,控制权返回进程被中断的那一点继续执行。

  (2)SIGIGN

  这个符号表示忽略该信号,执行了相应的signal()调用后,进程会忽略类型为sig的信号。

  (3)SIGDFL

  这个符号表示恢复系统对信号的默认处理。

  函数说明

  signal()会依参数signum 指定的信号编号来设置该信号的处理函数。当指定的信号到达时就会跳转到参数handler指定的函数执行。当一个信号的信号处理函数执行时,

  如果进程又接收到了该信号,该信号会自动被储存而不会中断信号处理函数的执行,直到信号处理函数执行完毕再重新调用相应的处理函数。但是如果在信号处理函数执行时进程收到了其它类型的信号,该函数的执行就会被中断。

  返回值: 返回先前的信号处理函数指针,如果有错误则返回SIG_ERR(-1)。

  附加说明 :在信号发生跳转到自定的handler处理函数执行后,系统会自动将此处理函数换回原来系统预设的处理方式,如果要改变此操作请改用sigaction()。

  下面的情况可以产生Signal:

  1. 按下CTRL+C产生SIGINT

  2. 硬件中断,如除0,非法内存访问(SIGSEV)等等

  3. Kill函数可以对进程发送Signal

  4. Kill命令。实际上是对Kill函数的一个包装

  5. 软件中断。如当Alarm Clock超时(SIGURG),当Reader中止之后又向管道写数据(SIGPIPE),等等

  程序例:

  /* This example installs a signal handler routine for SIGFPE,

  catches an integer overflow condition, makes an adjustment

  to AX register, and returns. This example program MAY cause

  your computer to crash, and will produce runtime errors

  depending on which memory model is used.

  */

  #pragma inline

  #include <stdio.h>

  #include <signal.h>

  void Catcher(int sig, int type, int *reglist)

  {

  printf("Caught it!/n");

  *(reglist + 8) = 3; /* make return AX = 3 */

  }

  int main(void)

  {

  signal(SIGFPE, Catcher);

  asm mov ax,07FFFH /* AX = 32767 */

  asm inc ax /* cause overflow */

  asm into /* activate handler */

  /* The handler set AX to 3 on return. If that hadn't happened,

  there would have been another exception when the next 'into'

  was executed after the 'dec' instruction. */

  asm dec ax /* no overflow now */

  asm into /* doesn't activate */

  return 0;

  }

  2 Signals:

  

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专用,进程超过资源限制的时候发送

注意点

  1、不要使用低级的或者STDIO.H的IO函数

  2、不要使用对操作

  3、不要进行系统调用

  4、不是浮点信号的时候不要用longjmp

  5、singal函数是由ISO C定义的。因为ISO C不涉及多进程,进程组以及终端I/O等,所以他对信号的定义非常含糊,以至于对UNIX系统而言几乎毫无用处。

  备注:因为singal的语义于现实有关,所以最好使用sigaction函数替代本函数。

扩展阅读:
  • 1

    sigaction,kill,raise,Unix信号

`signal.signal` 是 Python `signal` 模块中的一个重要函数,用于设置信号处理程序。 ### 使用方法 `signal.signal` 函数的基本语法如下: ```python signal.signal(signalnum, handler) ``` - `signalnum`:这是一个整数,表示要捕获的信号编号,不同的信号编号对应不同的系统信号,例如 `signal.SIGINT` 通常代表用户按下 `Ctrl+C` 发送的中断信号。 - `handler`:这是一个可调用对象(如函数),或者是 `signal.SIG_IGN`、`signal.SIG_DFL` 或 `None` 中的一个。其中,`signal.SIG_IGN` 表示忽略该信号,`signal.SIG_DFL` 表示使用默认方式处理信号,`None` 表示先前的信号句柄没有设置。 以下是一个简单的示例,用于捕获 `SIGINT` 信号: ```python import signal def signal_handler(signum, frame): print(f'接收信号 {signum},程序即将退出') # 可以在这里添加更多的清理操作 import sys sys.exit(0) # 设置信号处理程序 signal.signal(signal.SIGINT, signal_handler) print('程序正在运行,按下 Ctrl+C 触发信号处理') while True: pass ``` 在上述代码中,定义了一个 `signal_handler` 函数作为信号处理程序,当接收到 `SIGINT` 信号,会执行函数。 ### 原理 实际上,`signal`、`pause`、`kill` 和 `alarm` 都是 Linux 应用编程中常见的 C 库函数,Python 只不过是用 Python 语言实现了这些功能。由于 Python 的解释器是使用 C 语言编写的,所以有此相似性并不意外。在 Python 3.4 中,`signal` 包被增强,信号阻塞等功能被加入到该包中 [^2]。 `signal.signal` 函数的返回值是对应于信号 `signalnum` 的信号句柄,可能是一个 Python 可调用对象,或者是 `signal.SIG_IGN`、`signal.SIG_DFL` 或 `None` 中的一个。这里 `signal.SIG_IGN` 意味着该信号此前被忽略,`signal.SIG_DFL` 意味着先前使用默认方式处理这个信号,`None` 意味着先前的信号句柄没有设置 [^1]。 当操作系统向 Python 进程发送一个信号,Python 解释器会检查该信号是否有对应的信号处理程序。如果有,则调用相应的处理程序;如果没有,则按照默认方式处理该信号
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值