我们在运行脚本的时候,使用Ctrl+c会直接终止,但我们希望做一些收尾工作,或者是我们在运行脚本时不希望被一些命令打断,这时候,我们就用到了trap命令。
trap命令用于指定在介绍到指定信号后将要采取的动作。比较常用的就是在终止脚本时执行清理工作。
Trap 有三种格式:
1:
trap "command" signal脚本收到signal-list清单内列出的信号时,trap命令执行双引号中的命令。
2:
Trap "commands"signal-list当脚本收到signal-list清单内列出的信号时,trap命令执行双引号中的命令
3:
Trap signal-list不指定任何命令,接受信号的默认操作,默认操作是结束进程的运行。
4:
Trap " "signal-listtrap命令指定一个空命令串,允许忽视信号,我们用到的就是这一种。
要注意的是:command只有一个是,不加引号也可以,在command外加单引号双引号是一样的。脚本程序通常是以从上到下的顺序解释执行的,所以必须在你想保护的那部分代码以前指定trap命令。所有信号都能捕捉 除了kill不能捕捉.
Linux信号简介
1) SIGHUP本信号在用户终端连接(正常或非正常)结束时发出,通常是在终端的控制进程结束时,通知同一session内的各个作业,这时它们与控制终端不再关联.
2) SIGINT程序终止(interrupt)信号,在用户键入INTR字符(通常是Ctrl-C)时发出
3) SIGQUIT和SIGINT类似,但由QUIT字符(通常是Ctrl-\)来控制.进程在因收到SIGQUIT退出时会产生core文件,在这个意义上类似于一个程序错误信号.
4) SIGILL执行了非法指令.通常是因为可执行文件本身出现错误,或者试图执行数据段.堆栈溢出时也有可能产生这个信号.
5) SIGTRAP由断点指令或其它trap指令产生.由debugger使用.
6) SIGABRT程序自己发现错误并调用abort时产生.
7) SIGIOT在PDP-11上由iot指令产生,在其它机器上和SIGABRT一样.
8) SIGBUS非法地址,包括内存地址对齐(alignment)出错. eg:访问一个四个字长的整数,但其地址不是4的倍数.
9) SIGFPE在发生致命的算术运算错误时发出.不仅包括浮点运算错误,还包括溢出及除数为0等其它所有的算术的错误.
10) SIGKILL用来立即结束程序的运行.本信号不能被阻塞,处理和忽略.
11) SIGUSR1留给用户使用
12) SIGSEGV试图访问未分配给自己的内存,或试图往没有写权限的内存地址写数据.
13) SIGUSR2留给用户使用
14) SIGPIPEBroken pipe
15) SIGALRM时钟定时信号,计算的是实际的时间或时钟时间. alarm函数使用该信号.
16) SIGTERM程序结束(terminate)信号,与SIGKILL不同的是该信号可以被阻塞和处理.通常用来要求程序自己正常退出. shell命令kill缺省产生这个信号.
17) SIGCHLD子进程结束时,父进程会收到这个信号.
18) SIGCONT让一个停止(stopped)的进程继续执行.本信号不能被阻塞.可以用一个handler来让程序在由stopped状态变为继续执行时完成特定的工作.例如,重新显示提示符
19) SIGSTOP停止(stopped)进程的执行.注意它和terminate以及interrupt的区别:该进程还未结束,只是暂停执行.本信号不能被阻塞,处理或忽略.
20) SIGTSTP停止进程的运行,但该信号可以被处理和忽略.用户键入SUSP字符时(通常是Ctrl-Z)发出这个信号
21) SIGTTIN当后台作业要从用户终端读数据时,该作业中的所有进程会收到SIGTTIN信号.缺省时这些进程会停止执行.
22) SIGTTOU类似于SIGTTIN,但在写终端(或修改终端模式)时收到.
23) SIGURG有紧急数据或out-of-band数据到达socket时产生.
24) SIGXCPU超过CPU时间资源限制.这个限制可以由getrlimit/setrlimit来读取/改变
25) SIGXFSZ超过文件大小资源限制.
26) SIGVTALRM虚拟时钟信号.类似于SIGALRM,但是计算的是该进程占用的CPU时间.
27) SIGPROF类似于SIGALRM/SIGVTALRM,但包括该进程用的CPU时间以及系统调用的时间.
28) SIGWINCH窗口大小改变时发出.
29) SIGIO文件描述符准备就绪,可以开始进行输入/输出操作.
30) SIGPWRPower failure
例子:
我们写一个测试网络连接的ping命令脚本,使这个脚本能够接受Ctrl+c时停止ping动作并输出QUIT字符
#!/bin/bash
trap 'echo "QUIT" ; exit ' SIGINT
for I in {1..254}; do
ping -c1 -W2 192.168.200.$I &> /dev/null
if [ $? -eq 0 ];then
echo -e "^[[32m192.168.200.$I is up.^[[0m"
else
echo -e "\033[32m 192.168.200.$I is down.\033[0m"
fi
done