[转]Linux下的常见信号总结

转自 https://www.cnblogs.com/gaorong/p/6430905.html

在linux下有很多信号,按可靠性分为可靠信号和非可靠信号,按时间分为实时信号和非实时信号,linux进程也有三种方式来处理收到的信号:

(1)忽略信号,即对信号不做任何处理,其中,有两个信号不能忽略:SIGKILL及SIGSTOP;

(2)捕捉信号。定义信号处理函数,当信号发生时,执行相应的处理函数;

(3)执行缺省操作,Linux对每种信号都规定了默认操作。

Linux进程对实时信号的缺省反应是进程终止。但是对于高性能服务器编程来说,这是致命的缺陷,对于这类服务器需要保证在收到各种信号后仍然可以可靠运行,所以我们需要在理解各种信号的缘由和正确的处理方式。本文将笔者经常碰到的一些信号进行整理,结合自己的使用经验简要分析。

SIGHUP和控制台操作有关,当控制台被关闭时系统会向拥有控制台sessionID的所有进程发送HUP信号,默认HUP信号的action是 exit,如果远程登陆启动某个服务进程并在程序运行时关闭连接的话会导致服务进程退出,所以一般服务进程都会用nohup工具启动(该命令就是让忽略该信号)或写成一个 daemon(利用setsid进行)。

以下五组可以放在一块类比

SIGINT 终止进程,通常我们的Ctrl+C就发送的这个消息。

SIGQUIT 和SIGINT类似, 但由QUIT字符(通常是Ctrl- \ )来控制. 进程收到该消息退出时会产生core文件。

SIGKILL 消息编号为9,我们经常用kill -9来杀死进程发送的就是这个消息,程序收到这个消息立即终止,这个消息不能被捕获,封锁或这忽略,所以是杀死进程的终极武器。

SIGTERM 是不带参数时kill默认发送的信号,默认是杀死进程。

1.SIGINT SIGTERM区别

前者与字符ctrl+c关联,后者没有任何控制字符关联。

前者只能结束前台进程,后者则不是。

2.SIGTERM SIGKILL的区别

前者可以被阻塞、处理和忽略,但是后者不可以。

KILL命令的默认不带参数发送的信号就是SIGTERM.让程序有好的退出。

SIGTERM比较友好,进程能捕捉这个信号,根据您的需要来关闭程序。在某些情况下,假如进程正在进行作业而且不能中断,那么进程可以忽略这个SIGTERM信号。

因为SIGTERM可以被阻塞,所以有的进程不能被结束时,需要用kill发送SIGKILL。即:kill-9 进程号。

SIGSTOP 停止进程的执行,同SIGKILL一样不可以被应用程序所处理,注意它和terminate以及interrupt的区别:该进程还未结束, 只是暂停执行.由 Ctrl+Z 控制,用户可以使用使用fg/bg操作恢复执行前台或后台的进程。fg命令在前台恢复执行被挂起的进程,此时可以使用ctrl-z再次挂起该进程,bg命令在后台恢复执行被挂起的进程,而此时将无法使用ctrl-z再次挂起该进程。

SIGCONT 当SIGSTOP发送到一个进程时,通常的行为是暂停该进程的当前状态。如果发送SIGCONT信号,该进程将仅恢复执行。除了其他目的,SIGSTOP和SIGCONT用于Unix shell中的作业控制,无法捕获或忽略SIGCONT信号。

SIGPIPE 这个是向一个没有读进程的管道写数据产生的错误,这种解释过于官方。在网络编程中这个信号发生在如果客户端已经关闭了套接字, 而服务器调用了一次write,服务器就会收到一个RST segment,如果服务器再次调用write,这个时候就会产生SIGPIPE信号,系统默认的处理方式是关掉这个进程, 但是对于一个高可用的服务器程序来说,需要手动处理这个信号,所以你会看到许多服务器程序代码会在前面显式加上signal (SIGPIPE, SIG_IGN)来忽略这个信号。

SIGCHILD 这个同样是高性能服务器需要关注的信号,如果服务器采用fork产生的子进程推出后要调用wait进行资源回收,防止僵尸进程的产生,但是如果程序对子进程退出后的状态不感兴趣的话可以调用signal(SIGCHLD,SIG_IGN); 交给系统init去回收。子进程也不会产生僵尸进程了。

SIGSEGV 就是SegmentFault 试图访问未分配给自己的内存, 或试图往没有写权限的内存地址写数据,官方举得三个例子是:

  1. buffer overflow --- usually caused by a pointer reference out of range.  野指针
  2. stack overflow --- please keep in mind that the default stack size is 8192K.  栈溢出
  3. illegal file access --- file operations are forbidden on our judge system. 非法文件访问

SIGBUS 指针所对应的地址是有效地址,但总线不能正常使用该指针。通常是未对齐的数据访问所致。试图访问一块无文件内容对应的内存区域,比如超过文件尾的内存区域,或者以前有文件内容对应,现在为另一进程截断过的内存区域。apache爆出的一个bug就是mmap调用的时候文件发生改变就会爆出这个错误,具体参考这篇精彩的博客是 : 由mmap引发的SIGBUS

关于上面两个信号,《UNP》卷二中对于共享内存mmap引发这两个错误的原因用一张图描述:

SIGBUS意味着我们是在内存映射区内访问,但是已超出了底层支撑对象的大小。SIGSEGV则意味着我i们在内存映射区以外访问。

SIGURG I/O紧急信号,也就是tcp传输带外数据时使用,但是tcp手册 RFC6093中已经不建议使用紧急指针了,所以这个信号也就没什么用了。

SIGIO 当描述符上可以进行I/O时产生这个信号,这时五大IO模型中信号驱动IO模型的实现信号。

SIGALRM 时钟定时信号, 计算的是实际的时间或时钟时间.alarm函数使用该信号.

以上就是编程中常见到的信号,也是最引人关注的信号,至于其他信号,以后想起再写吧。

2018.6.14 append:

https://www.linuxjournal.com/article/10815 shell 捕获信号

原文地址:https://www.cnblogs.com/dirge/p/11614967.html

时间: 2024-10-12 02:59:07

[转]Linux下的常见信号总结的相关文章

linux下的常见信号总结

在linux下有很多信号,按可靠性分为可靠信号和非可靠信号,按时间分为实时信号和非实时信号,linux进程也有三种方式来处理收到的信号: (1)忽略信号,即对信号不做任何处理,其中,有两个信号不能忽略:SIGKILL及SIGSTOP: (2)捕捉信号.定义信号处理函数,当信号发生时,执行相应的处理函数: (3)执行缺省操作,Linux对每种信号都规定了默认操作. Linux进程对实时信号的缺省反应是进程终止.但是对于高性能服务器编程来说,这是致命的缺陷,对于这类服务器需要保证在收到各种信号后仍然

Linux下各种常见环境变量的配置

Linux系统下各种环境变量都通过修改/etc/profile文件来实现.由于是系统文件,修改此文件需要root权限.因此实现以下功能都需要用户拥有root权限. 另:不要轻易修改profile文件中的现有内容.应在保证原有内容不变的前提下,在文件的最后插入新的一行. 1.路径变量PATH 当用户在某工作目录执行一个命令或者可执行程序时,若当前路径下不存在该程序,则系统将从PATH路径列表中查找指定的应用程序.还记得java命令,ping命令么?这些应用程序所在的文件夹都包含在了PATH路径中.

Linux下的常见压缩解压缩命令

Linux常见压缩解压缩命令 常见压缩文件扩展名 .Z compress 程序压缩的文件: .zip zip 程序压缩的文件: .gz gzip 程序压缩的文件: .bz2 bzip2 程序压缩的文件: .xz xz 程序压缩的文件: .tar tar 程序打包的数据,并没有压缩过: .tar.gz tar 程序打包的文件,其中并且经过 gzip 的压缩 .tar.bz2 tar 程序打包的文件,其中并且经过 bzip2 的压缩 .tar.xz tar 程序打包的文件,其中并且经过 xz 的压缩

Linux下的signal信号机制

分享一下我老师大神的人工智能教程吧.零基础!通俗易懂!风趣幽默!还带黄段子!希望你也加入到我们人工智能的队伍中来!http://www.captainbed.net 在Linux中,要发送一个信号相当容易.程序员需要知道两个信息:要发送哪个信号,将这个信号发送给哪个进程.可以用 man 7 signal 找到一个可以利用的信号的列表.用户可以只将信号发送给用户自己的进程,也可以以root身份运行从而将信号发送给任意一进程. Source: #include<stdio.h> #include&

Linux下的常见命令

收集一些平时使用不多但却是很重要的命令,以后会持续更新...... 1. 查看当前Ubuntu系统是多少位的 uname -mi 打印结果:X86_64(64位),X86_32(32位) 2. 查看当前Ubuntu系统的版本号    cat /etc/issue 打印结果:Ubuntu 12.04 LTS \n \l 3. 查看系统硬盘和外部挂载磁盘的使用 Filesystem      Size  Used Avail Use% Mounted on /dev/sda1       909G

LINUX下的几个常见的环境变量

环境变量是指操作系统用来指定操作系统运行环境的一些操作,linux下的常见的几个环境变量如下: PATH              指定命令的搜索路径 HOME              当前用户的主目录 HISTSIZE          历史命令的记录条数 LOGNAME           当前用户的登录名 HOSTNAME          主机名称 SHELL             当前用户的shell类型 LANG/LANGUGE      语言相关的环境变量 查看方式可通过ech

Linux下利用signal函数处理ctrl+c等信号

前言 linux下可以通过信号机制来实现程序的软中断,是一个非常有用的编程方法.我们平时在程序运行的时候按下ctrl-c.ctrl-z或者kill一个进程的时候其实都等效于向这个进程发送了一个特定信号,当进程捕获到信号后,进程会被中断并立即跳转到信号处理函数.默认情况下一个程序对ctrl-c发出的信号(SIGINT)的处理方式是退出进程,所以当我们按下ctrl-c的时候就可以终止一个进程的运行. signal函数 但是有时候我们希望我们的程序在被信号终止之前执行一些特定的收尾流程,或者我们希望我

Linux下信号

信号和 中断是很类似的,只不过是一个是硬件中断,另外一个是软中断.中断是系统对于异步事件的响应. 简单理解就是:中断源 发出 中断信号 在 中断向量表 中执行 中断处理程序 之前保存 现场信息 异步事件的响应:进程执行代码的过程中可以随时被打断,然后去执行异常处理程序. 中断源发出中断信号,CPU判断中断是否屏蔽屏蔽.保护现场 ,cpu执行中断处理程序, cpu恢复现场,继续原来的任务. 中断向量表保存了中断处理程序的入口地址. 中断个数固定,操作系统启动时初始化中断向量表. 中断有优先级,中断

Unix / Linux 下 nohup 和 &amp; 的区别

声明:本文首发 简单教程,网址为 https://www.twle.cn/t/332#reply0 就在刚刚回家的路上,被前同事夺命三连 call 呼唤解决一个问题:为啥放在 crontab 里的命令放在 Shell 会进入假死状态? 那我就问了?什么是假死状态? 同事说,就是一直不会执行完毕,占着 Shell,不能做其它事情. 多次沟通后,才知道他写的这个是一个守护进程似的死循环程序,一旦启动除非发生意外,否则不会自己退出. 那肯定,很显然,会占着 Shell 啊. nohup 和 & 我给的