shell脚本trap捕获信号教程详解

发布时间:2020-03-09编辑:脚本学堂
有关shell脚本中用trap捕获信号的方法,在shell脚本编程中信号捕获很重要,可以用来处理一些特殊操作,trap命令就是用来做这个的,一起了解下。

linux shell/ target=_blank class=infotextkey>shell脚本中,可以用trap命令捕获信号,包括中断信号,退出信号等,本节就对这一命令trap进行详细介绍。

信号是一种进程间通信机制,它给应用程序提供一种异步的软件中断,使应用程序有机会接受其他程序活终端发送的命令(即信号)。

应用程序收到信号后,有三种处理方式:忽略,默认,或捕捉。进程收到一个信号后,会检查对该信号的处理机制。

如果是SIG_IGN,就忽略该信号;如果是SIG_DFT,则会采用系统默认的处理动作,通常是终止进程或忽略该信号;
如果给该信号指定了一个处理函数(捕捉),则会中断当前进程正在执行的任务,转而去执行该信号的处理函数,返回后再继续执行被中断的任务。

在有些情况下,不希望shell脚本在运行时刻被中断,比如shell脚本设为某一用户的默认shell,使这一用户进入系统后只能作某一项工作,当数据备份时当然不希望用户使用Ctrl c之类便进入到shell状态,此时便需要信号处理的相关知识。

一些常见信号:
 

信号名称  信号数   描述
SIGHUP  1 本信号在用户终端连接(正常或非正常)结束时发出, 通常是在终端的控制进程结束时, 通知同一session内的各个作业, 这时它们与控制终端不再关联。 登录Linux时,系统会分配给登录用户一个终端(Session)。在这个终端运行的所有程序,包括前台进程组和后台进程组,一般都属于这个Session。当用户退出Linux登录时,前台进程组和后台有对终端输出的进程将会收到SIGHUP信号。这个信号的默认操作为终止进程,因此前台进程组和后台有终端输出的进程就会中止。
对于与终端脱离关系的守护进程,这个信号用于通知它重新读取配置文件。
SIGINT  2 程序终止(interrupt)信号, 在用户键入INTR字符(通常是Ctrl C)时发出。
SIGQUIT  3 和SIGINT类似, 但由QUIT字符(通常是Ctrl /)来控制. 进程在因收到SIGQUIT退出时会产生core文件, 在这个意义上类似于一个程序错误信号。
SIGFPE  8 在发生致命的算术运算错误时发出. 不仅包括浮点运算错误, 还包括溢出及除数为0等其它所有的算术的错误。
SIGKILL  9 用来立即结束程序的运行. 本信号不能被阻塞, 处理和忽略。
SIGALRM  14 时钟定时信号, 计算的是实际的时间或时钟时间. alarm函数使用该信号。
SIGTERM  15 程序结束(terminate)信号, 与SIGKILL不同的是该信号可以被阻塞和处理. 通常用来要求程序自己正常退出. shell命令kill缺省产生这个信号。

捕获信号
当按下Ctrl + C键或break键在终端一个shell程序的执行过程中,正常程序将立即终止,并返回命令提示符。
这可能并不总是可取的。

例如,可能最终留下了一堆临时文件,将不会清理。

捕获这些信号是很容易的,trap命令的语法:
 

$ trap commands signals
 

命令可以是任何有效的linux命令,或一个用户定义的函数,信号可以是任意数量的信号,想来捕获的列表。

在shell脚本中的陷阱有三种常见的用途:
1、清理临时文件
2、忽略信号
3、清理临时文件:
trap命令,可以删除一些文件,然后退出,如果有人试图从终端中止程序:
 

trap "rm -f $WORKDIR/work1$$ $WORKDIR/dataout$$; exit" 2
 

执行shell程序,这个陷阱的角度,这两个文件work1$$ 和 dataout$$将被自动删除,如果程序接收信号数为2。

因此,用户中断执行,如果执行的程序后,这个陷阱你可以放心,这两个文件将被清理。
exit 命令如下 rm 是必要的,因为没有它的执行将继续在节目中的一点,它离开时收到信号。
信号产生挂断:要么有人故意挂断线路或线路被意外断开。

可以修改前面的陷阱也删除指定的文件,在这种情况下,两个信号信号1号添加到列表:
 

$ trap "rm $WORKDIR/work1$$ $WORKDIR/dataout$$; exit" 1 2
 

现在,这些文件将被删除,如果该行被挂了,或按Ctrl c键被按下。

来捕获指定的命令必须用引号括起来,如果它们包含一个以上的命令。

注意,在 shell 命令行扫描 trap 命令得到执行,并再次当一个所列出的的信号被接收的时间。
WORKDIR 值 $$ 所以在前面的例子中,将被取代 trap 命令执行的时间。

如果想这种替代发生在收到信号1或2的时间,可以把单引号内的命令:
 

$ trap 'rm $WORKDIR/work1$$ $WORKDIR/dataout$$; exit' 1 2

忽略信号:
如果陷阱列出的命令是空的,指定的信号接收时,将被忽略。

例如,命令:
 

$ trap '' 2

指定的中断信号是被忽略的。
可能要忽略某些信号时进行一些操作,不希望打断。

可以指定多个信号被忽略:
 

$ trap '' 1 2 3 15

注意,第一个参数必须被指定为一个信号被忽略,而不是相当于写入下面的内容,它具有独立的含义也各有:
 

$ trap  2

如果忽略了一个信号,所有的子shell也忽略该信号。
不过,如果指定要采取的行动在收到的信号,所有的子shell仍然会在收到该信号的默认操作。

重设陷阱:
当改变了默认在收到信号后应采取的动作,可以改变它回来的陷阱,如果你只是省略第一个参数;
 

$ trap 1 2

复位应采取的动作收到信号1或2 返回默认。