十三、信号

本文详细介绍了Linux系统中的信号机制,包括信号的基本概念、产生条件、处理方式,以及如何修改信号响应方式。通过实例解析了如何处理中断信号、僵死进程,以及如何使用kill和raise函数发送不同类型的信号。通过对信号的理解和应用,读者可以更好地掌握进程管理和异常处理。
摘要由CSDN通过智能技术生成

一、信号基本概念

信号是系统预先定义好的某些特定时间,信号可以被产生,也可以被接收,产生和接收的实体都是进程,信号的作用就是一个进程向另一个进程通知某一时间的发生;是一种软件中断,提供一种处理异步事件的方法,如中断输入中断CTRL C,则会通过信号机制暂停一个程序,或及早终止管道中的下一个程序。
信号的名字都是以SIG开头,信号的定义文件位置在:

/usr/include/bits/signum.h

文件内容:
在这里插入图片描述
在头文件<signal.h>中一种事件一个信号,被定义为宏值,用正整数表示,不存在编号为0的信号,因为kill函数对信号编号0有特殊应用,在kill部分我们进行讲解,我们也可以用:

kill -l //查看所有的信号

在这里插入图片描述

(一)信号的产生

产生信号需要一定的条件,有几种常见的产生信号的条件:

  • 当用户按某些终端键时,引发终端产生的信号。如在终端桑按DELETE键,或Ctrl+C通常会产生中断信号SIGINT。这时停止一个已失去控制的程序的方法。
  • 硬件异常产生的信号,除数为0,无效地址等。这些条件通常由硬件检测到,并将其通知内核,然后内核为该条件发生时正在运行的进程产生适当的信号,如对执行一个无效内存引用的进程产生SIGSEGV信号。
  • 进程调用kill(2)函数可以将信号发送给另一个进程或进程组,自然,对此有所限制,接受信号进程和发送信号进程的所有者必须相同,或者发送信号进程的所有者必须是超级用户。
  • 用户可用kill(1)可将信号发送给其他进程,此命令只是kill函数的接口,常用此命令终止一个失控的后台进程。
  • 当检测到某种软件条件发生时,并应将其通知有关进程时也产生信号,如SIGURG网络连接上传来带外数据时产生。

信号事件对于进程来说是随机出现的,进程不能简单的根据变量派判断是否出现了一个信号,而是必须告诉内核,在此信号出现时,请执行下列操作,那么内核在某个信号出现时如何处理呢?

(二)信号的处理

主要有这三种方式,我们称为信号的处理或信号相关的动作:

  • 忽略此信号。大多数信号都可以使用这种方式进程处理,但又两种信号绝不能被忽略,它们是SIGKILL(9号信号),SIGSTOP(15号信号),原因是:它们像超级用户提供了使进程终止或停止的可靠办法。如果忽略某些由硬件异常产生的信号,则进程的运行行为是未定义的。
  • 捕捉此信号,要通知内核在某种信号发生时调用一个用户函数,在用户函数中,可执行用户希望对这种事件进行的处理。例如,若正在运行一个命令解释器,当用户键盘产生中断信号时,希望该命令解释器返回到主循环,终止正在为该用户执行的命令。如果捕捉到SIGCHLD信号,表示子进程已经终止,此信号的捕捉函数可以调用waitpid取得该子进程的进程ID和终止状态,进行僵死进程处理。不能捕捉SIGKILL,SIGSTOP信号。
  • 执行系统的默认动作,这些都是系统规定的。

(三)信号的种类&常见信号

将信号分为可靠信号(实时信号)和不可靠信号(非实时信号):

  • 可靠信号:Linux改进了信号机制,新增了32种信号,均为可靠信号,信号支持排队,不会丢失,发多少次,就可以收到多少次,信号值位于[SIGRTMIN,SIGRTMAX]区间都是可靠信号。
  • 非可靠信号:从unix系统继承过来的信号都是非可靠信号,信号不支持排队,可能会丢失,比如发送多次相同的信号,进程只能收到一次,信号值小于SIGRTMIN的都是非可靠信号。

可以通过kill -l查看信号的种类,我们需要了解常见的几种信号:

信号 宏值整数 含义
SIGHUP 1 终端接口检测到一个连接断开,则将此信号发送给与终端相关的控制进程
SIGINT 2 当用户按中断键(delete,Ctrl+c)时,终端驱动程序产生此信号并送至前台进程组的每一个进程,当一个进程运行失控时,常用此信号终止它。
SIGQUIT 3 当用户在终端上按退出键(Ctrl+/)时产生此信号,并送至前台进程组中的所有进程,会终止进程组,还会产生一个core文件。
SIGTERM 15 由kill命令发送的系统默认终止信号
SIGKILL 9 不能被忽略和捕捉的信号,为管理员提供了一种可以杀死任一进程的方法。所以kill -9 pid可以杀死任何进程。
SIGCONT 18 让进程继续运行,和STOP相反,就是bg命令,将挂起的进程发到后台执行
SIGSTOP 19 停止一个进程,按Ctrl+z键会触发。
SIGCHLD 17 当一个进程终止或停止时,将此信号发送给父进程,系统默认忽略,我们可以设置捕捉此信号,进行wait处理僵死进程。

二、修改信号响应方式的signal函数

(一)概念

signal函数时信号机制最简单的接口,函数原型如下:

# include<signal.h>
void (*signal(int signo,void (*func)(int)))(int);

参数说明:

  • signo是信号名;
  • func的值是常量SIG_IGN,SIG_DFL,或捕捉到此信号后要调用的函数地址。如果是SIG_IGN,内核忽略此信号;如果是SIG_DFL,则表示此信号按照系统默认处理方式处理;如果是函数地址时,表示在信号发生时,调用该函数,该函数的格式必须是:一个整型参数,无返回值。
  • signal函数的返回值是一个函数地址,就是处理信号函数的地址,处理信号函数有一个整型参数,就是后面的(int),即signal需要向信号处理程序传送一个整型参数,这个参数就是信号的正整数宏,处理函数无返回值。

(二)实例1:修改中断信号的响应方式

题目: 实现对中断信号的捕捉,修改响应方式,当我们进行Ctrl c 时程序不会中断,会输出"Hello Linux!"和signal传过来的参数sign。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值