@orangleliu
2016-03-04T09:36:11.000000Z
字数 2598
阅读 3025
Linux
信号是Unix系统中常见的一种进程间通信方式(IPC),例如我们经常操作的
kill -9 pid这里的-9对应的就是 SIGKILL 信号, 9就是这个信号的编号,SIGKILL是它的名称。 由于不同版本的 *nux 的实现会有差异,具体请参照系统API,我这里是OSX,可以使用man signal查看所有信号的定义。这里简单的学习下Python标准库中信号处理的模块 signal模块.
由于signal是系统编程的接口,那么咱们来看看他的概念。
信号(signal)是Linux进程间通信的一种机制,全称为软中断信号,也被称为软中断。信号本质上是在软件层次上对硬件中断机制的一种模拟。
再来看看常用的场景
与其他进程间通信方式(例如管道、共享内存等)相比,信号所能传递的信息比较粗糙,只是一个整数。但正是由于传递的信息量少,信号也便于管理和使用,可以用于系统管理相关的任务,例如通知进程终结、中止或者恢复等。每种信号用一个整型常量宏表示,以SIG开头,比如SIGCHLD、SIGINT等,它们在系统头文件中定义。
更具体的介绍和详细的机制原理,请参考 《Unix环境高级编程》等书籍。
自己的理解就是:可以给一个正在运行的进程发送不同的信号,然后进程就能立即收到这个通知,并且做出响应的行为。
常用的几个信号
| 编号 | 名称 | 作用 |
|---|---|---|
| 1 | SIGHUP | 终端挂起或者终止进程。默认动作为终止进程 |
| 2 | SIGINT | 键盘中断 <ctrl+c> 经常会用到。默认动作为终止进程 |
| 3 | SIGQUIT | 键盘退出键被按下。一般用来响应 <ctrl+d>。 默认动作终止进程 |
| 9 | SIGKILL | 强制退出。 shell中经常使用 |
| 14 | SIGALRM | 定时器超时,默认为终止进程 |
| 15 | SIGTERM | 程序结束信号,程序一般会清理完状态在退出,我们一般说的优雅的退出 |
在文档的开头,讲述了Python signal对于系统的封装和一些使用常识, 使用之前应当认真阅读一下。
signal.signal(signalnum, handler) 针对不同的信号需要定义对应的处理函数,当运行中的程序接受到对应的信号时候,会调用对应的handler。 handler函数应当有2个参数,一个是 signalnum, 另一是 stack frame(None 或者是 frame对象)例如写一个小程序,来处理 ctrl+c事件和 SIGHUP,也就是1和2信号。
#coding:utf-8#orangleliu py2.7#recv_signal.pyimport signalimport timeimport sysimport osdef handle_int(sig, frame):print "get signal: %s, I will quit"%sigsys.exit(0)def handle_hup(sig, frame):print "get signal: %s"%sigif __name__ == "__main__":signal.signal(2, handle_int)signal.signal(1, handle_hup)print "My pid is %s"%os.getpid()while True:time.sleep(3)
我们来测试下,首先启动程序(根据打印的pid),在另外的窗口输入 kill -1 21838 和 kill -HUP 21838, 最后使用 ctrl+c关闭程序。 程序的输出如下:
# python recv_signal.pyMy pid is 21838get signal: 1get signal: 1^Cget signal: 2, I will quit
signal.getsignal(signalnum) 根据signalnum返回信号对应的handler,可能是一个可以调用的Python对象,或者是 signal.SIG_IGN(表示被忽略), signal.SIG_DFL(默认行为已经被使用) 或 None(Python的handler还没被定义)。获取signal中定义的信号num和名称,还有它的handler是什么
#coding:utf-8#orangleliu py2.7#getsignal_handler.pyimport signaldef handle_hup(sig, frame):print "get signal: %s"%sigsignal.signal(1, handle_hup)if __name__ == "__main__":ign = signal.SIG_IGNdfl = signal.SIG_DFLprint "SIG_IGN", ignprint "SIG_DFL", dflprint "*"*40for name in dir(signal):if name[:3] == "SIG" and name[3] != "_":signum = getattr(signal, name)gsig = signal.getsignal(signum)print name, signum, gsig
运行的结果:可以看到大部分信号都是都有默认的行为。
SIG_IGN 1SIG_DFL 0****************************************SIGABRT 6 0SIGALRM 14 0SIGBUS 10 0SIGCHLD 20 0SIGCONT 19 0SIGEMT 7 0SIGFPE 8 0SIGHUP 1 <function handle_hup at 0x109371c80>SIGILL 4 0SIGINFO 29 0SIGINT 2 <built-in function default_int_handler>SIGIO 23 0SIGIOT 6 0SIGKILL 9 NoneSIGPIPE 13 1SIGPROF 27 0SIGQUIT 3 0SIGSEGV 11 0SIGSTOP 17 NoneSIGSYS 12 0SIGTERM 15 0SIGTRAP 5 0SIGTSTP 18 0SIGTTIN 21 0SIGTTOU 22 0SIGURG 16 0SIGUSR1 30 0SIGUSR2 31 0SIGVTALRM 26 0SIGWINCH 28 0SIGXCPU 24 0SIGXFSZ 25 1