Signal ()函数详细介绍

本文详细介绍了signal()函数的使用方法及参数含义,并通过实例演示了如何处理不同类型的信号。此外,还列举了一些常用信号及其作用。

signal()函数理解

<signal.h> 这个头文件中。

signal(参数1,参数2);

参数1:我们要进行处理的信号。系统的信号我们可以再终端键入 kill -l查看(64)。其实这些信号时系统定义的宏。

参数2:我们处理的方式(是系统默认还是忽略还是捕获)。

一般有3中方式进行操作。

1eg: signal(SIGINT ,SIG_ING );

//SIG_ING 代表忽略SIGINT信号SIGINT信号代表由InterruptKey产生,通常是CTRL +C 或者是DELETE 。发送给所有ForeGround Group的进程。

下面我们写个死循环:


这时我们保存执行。

按下CTRL _C程序没有反应。这就对了

如果我们想结束该程序可以按下CTRL +\来结束

其实当我们按下CTRL +\组合键时,是产生了SIGQUIT信号

2eg: signal(SIGINT ,SIG_DFL );

//SIGINT信号代表由InterruptKey产生,通常是CTRL +C或者是DELETE。发送给所有ForeGroundGroup的进程。SIG_DFL代表执行系统默认操作,其实对于大多数信号的系统默认动作时终止该进程。这与不写此处理函数是一样的。

我们将上面的程序改成


这时就可以按下CTRL +C 来终止该进程了。把signal(SIGINT,SIG_DFL);这句去掉,效果是一样的。

(3) void ( *signal( int sig, void (* handler)( int )))( int );
int (*p)();
这是一个函数指针, p所指向的函数是一个不带任何参数, 并且返回值为int的一个函数.
int (*fun())();
这个式子与上面式子的区别在于用fun()代替了p,而fun()是一个函数,所以说就可以看成是fun()这个函数执行之后,它的返回值是一个函数指针,这个函数指针(其实就是上面的p)所指向的函数是一个不带任何参数,并且返回值为int的一个函数.

void (*signal(int signo, void (*handler)(int)))(int);就可以看成是signal()函数(它自己是带两个参数,一个为整型,一个为函数指针的函数),而这个signal()函数的返回值也为一个函数指针,这个函数指针指向一个带一个整型参数,并且返回值为void的一个函数.

在写信号处理函数时对于信号处理的函数也是void sig_fun(int signo);这种类型,恰好与上面signal()函数所返回的函数指针所指向的函数是一样的.void ( *signal() )( int );

signal是一个函数, 它返回一个函数指针, 后者所指向的函数接受一个整型参数 且没有返回值, 仔细看, 是不是siganal( int signo, void (*handler)(int) )的第2个参数了,对了,其实他所返回的就是 signal的第2个信号处理函数,指向信号处理函数,就可以执行函数了( signal内部时, signal把信号做为参数传递给handler信号处理函数,接着 signal函数返回指针, 并且又指向信号处理函数, 就开始执行它)

那么,signal函数的参数又是如何呢?signal函数接受两个参数:一个整型的信号编号,以及一个指向用户定义的信号处理函数的指针。我们此前已经定义了指向用户定义的信号处理函数的指针sfp:

void (*sfp)(int);

sfp 的类型可以通过将上面的声明中的sfp去掉而得到,即void (*)(int)。此外,signal函数的返回值是一个指向调用前的用户定义信号处理函数的指针,这个指针的类型与sfp指针类型一致。因此,我们可以如下声明signal函数:

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

同样地,使用typedef可以简化上面的函数声明:

typedef void (*HANDLER)(int);
HANDLER signal(int, HANDLER);

Ok;看个例子:


此程序是对当我们按下CTRL +C键时,会执行我们定义的信号处理函数。

每当我们按下CTRL +C键时会打印该信号的number.可以看出该信号的num2

要想退出可以按下CTRL +\ 打印结果为最后一行。

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

转载:http://blog.youkuaiyun.com/ta893115871/article/details/7475095

### signal 函数的用法及实现 在 C 语言中,`signal` 函数用于捕获并处理异步信号。它允许程序定义当特定信号发生时应执行的操作。以下是 `signal` 函数的基本信息及其 C 和 Python 的实现方式。 #### C 语言中的 `signal` 函数 `signal` 函数在 `<signal.h>` 头文件中声明,其原型如下: ```c typedef void (*sighandler_t)(int); sighandler_t signal(int signum, sighandler_t handler); ``` - 参数 `signum` 表示需要捕获的信号编号。 - 参数 `handler` 是一个函数指针,指向信号处理函数。如果将 `handler` 设置为 `SIG_IGN`,则忽略该信号;如果设置为 `SIG_DFL`,则恢复默认行为[^4]。 当指定信号发生时,程序会调用 `handler` 指定的函数。以下是一个简单的示例,展示如何使用 `signal` 函数捕获 `SIGINT`(通常由 Ctrl+C 触发)信号: ```c #include <stdio.h> #include <stdlib.h> #include <signal.h> #include <unistd.h> void sigint_handler(int signum) { printf("Caught signal %d\n", signum); } int main() { // 设置 SIGINT 的处理函数 signal(SIGINT, sigint_handler); while (1) { printf("Waiting for signal...\n"); sleep(1); // 程序挂起一秒 } return 0; } ``` 在上述代码中,当用户按下 Ctrl+C 时,程序不会终止,而是调用 `sigint_handler` 函数输出一条消息[^4]。 #### Python 中的 `signal` 模块 Python 提供了 `signal` 模块,用于处理信号。与 C 类似,可以通过定义回调函数来捕获和响应信号。以下是一个使用 Python 捕获 `SIGINT` 的示例: ```python import signal import time def sigint_handler(signum, frame): print(f"Caught signal {signum}") # 设置 SIGINT 的处理函数 signal.signal(signal.SIGINT, sigint_handler) while True: print("Waiting for signal...") time.sleep(1) ``` 在 Python 中,`signal.signal()` 方法的第一个参数是信号编号,第二个参数是处理函数或常量 `signal.SIG_IGN` 或 `signal.SIG_DFL`。运行此脚本后,按下 Ctrl+C 将触发自定义处理函数[^4]。 #### 实现细节与注意事项 - 在多线程环境中,`signal` 的行为可能受到限制。例如,在 POSIX 标准下,只有主线程可以设置信号处理程序。 - 信号处理函数应尽量简短,避免执行复杂的操作,因为信号可能会中断任意代码段的执行。 ###
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值