现在的位置: 首页 > 综合 > 正文

信号trap的应用(转)

2014年02月06日 ⁄ 综合 ⁄ 共 3185字 ⁄ 字号 评论关闭
文章目录

一. 信号简介

在有些情况下,我们不希望自己的shell脚本在运行时刻被中断,比如说我们写得shell脚 本设为某一用户的默认shell,使这一用户进入系统后只能作某一项工作,如数据库备份, 我们可不希望用户使用ctrl+C之类便进入到shell状态,做我们不希望做的事情。这便用到了信号.
处理。

二. 列出信号

kill -l可以列出系统的信号名称,如下:

#kill -l

也可以使用下面的命令:

#man signal

1) SIGHUP 2) SIGINT 3) SIGQUIT 4) SIGILL
5) SIGTRAP 6) SIGABRT 7) SIGBUS 8) SIGFPE
9) SIGKILL 10) SIGUSR1 11) SIGSEGV 12) SIGUSR2
13) SIGPIPE 14) SIGALRM 15) SIGTERM 17) SIGCHLD
18) SIGCONT 19) SIGSTOP 20) SIGTSTP 21) SIGTTIN
22) SIGTTOU 23) SIGURG 24) SIGXCPU 25) SIGXFSZ
26) SIGVTALRM 27) SIGPROF 28) SIGWINCH 29) SIGIO
30) SIGPWR 31) SIGSYS 34) SIGRTMIN 35) SIGRTMIN+1
36) SIGRTMIN+2 37) SIGRTMIN+3 38) SIGRTMIN+4 39) SIGRTMIN+5
40) SIGRTMIN+6 41) SIGRTMIN+7 42) SIGRTMIN+8 43) SIGRTMIN+9
44) SIGRTMIN+10 45) SIGRTMIN+11 46) SIGRTMIN+12 47) SIGRTMIN+13
48) SIGRTMIN+14 49) SIGRTMIN+15 50) SIGRTMAX-14 51) SIGRTMAX-13
52) SIGRTMAX-12 53) SIGRTMAX-11 54) SIGRTMAX-10 55) SIGRTMAX-9
56) SIGRTMAX-8 57) SIGRTMAX-7 58) SIGRTMAX-6 59) SIGRTMAX-5
60) SIGRTMAX-4 61) SIGRTMAX-3 62) SIGRTMAX-2 63) SIGRTMAX-1
64) SIGRTMAX

三. 发送信号

语法: kill -signal pid

signal可以是要发送的信号的整数,与可以是信号的名称.

pid是信号要发送的进程PID,如:

kill -s SIGTERM pid    #同killl pid

向进程号为1001的进程发送1号信号

kill -1 1001

向进程号为1001的进程发送SIGQUIT信号

kill -s SIGQUIT 1001

四. 信号处理

(一). trap捕捉到信号之后,可以有三种反应方式:

  1. 执行一段程序来处理这一信号

  2. 接受信号的默认操作

  3. 忽视这一信号

(二). trap对上面三种方式提供了三种基本形式:

  第一种形式的trap命令在shell接收到signal list清单中数值相同的信号时,将执行双

  引号中的命令串。

  trap 'commands' signal-list

  trap "commands" signal-list

  为了恢复信号的默认操作,使用第二种形式的trap命令:

  trap signal-list

  第三种形式的trap命令允许忽视信号

  trap " " signal-list

  注意:

  1. 对信号11(段违例)不能捕捉,因为shell本身需要捕捉该信号去进行内存的转储。

  2. 在trap中可以定义对信号0的处理(实际上没有这个信号), shell程序在其终止(如

  执行exit语句)时发出该信号。

  3. 在捕捉到signal-list中指定的信号并执行完相应的命令之后, 如果这些命令没有

  将shell程序终止的话,shell程序将继续执行收到信号时所执行的命令后面的命令,这样将

  很容易导致shell程序无法终止。

  另外,在trap语句中,单引号和双引号是不同的,当shell程序第一次碰到trap语句时,

  将把commands中的命令扫描一遍。此时若commands是用单引号括起来的话,那么shell不会

  对commands中的变量和命令进行替换, 否则commands中的变量和命令将用当时具体的值来

例9-1 信号的三种处理方式

在信号1(挂起),信号2(中断),信号15(软件中止)中任何一个到达时,删除所有临时文件,然后退出.

1. #trap 'rm tmp*;exit 1' 0 1 2 15 #或trap 'rm tmp*;exit 1' EXIT
HUP
INT
TERM

将信号2重置为按缺省方式处理

2. #trap 2 #或trap INT

忽略信号1,信号2

3. #trap " " 1 2 #或trap ""
HUP INT

列出中断信号与键盘的对应

4. stty -a

例9-2 禁止用户按下退出键退出程序

trap 'echo "control-C will not terminate $0."'
INT

trap 'echo "control-\ will not terminate $0."' QUIT
trap 'echo "control-Z will not terminate $0."' TSTP

echo "enter any string after the prompt,when you are ready to exit ,type \"stop\". "

while :
do
   echo -n "Go ahead ...>"
   read
   if [[ $REPLY == [Ss]top ]] ;then
     break
   fi
done

例9-3 在函数中捕捉信号

function trapper() {
   echo "In trapper"
   trap 'echo "Caught in a trap"'
INT

}

while :
do
   echo "In the main script"
   trapper
   echo "Still in main"
   sleep 5
done

例9-4 退出时,删除临时文件夹

当脚本接收到信号SIGHUP,SIGINT,SIGQUIT,SIGTERM,临时文件夹就会被删除,并且调用exit以返回值2退出.

cleanup() {
     if [ -f "$OUTFILE" ] ;then
         printf "Cleaning up_";
         rm -f "$OUTFILE" 2> /dev/null ;
         echo "Done";
     fi
}

trap cleanup 1 2 3 15

例9-5 定时器脚本

#!/bin/bash

AlarmHandler() {
echo "Got SIGALARM,cmd took too long."
KillSubProcs
exit 14
}

KillSubProcs() {
kill ${CHPROCIDS:-$!}
if [ $? -eq 0 ] ;then
echo "Sub-processes killed";
fi
}

SetTimer() {
DEF_TOUT=${1:-10};
if [ $DEF_TOUT -ne 0 ] ;then
sleep $DEF_TOUT && kill -s 14 $$ &
CHPROCIDS=$!
TIMERPROC=$!
fi
}

UnsetTimer() {
kill $TIMERPROC
}

#main

trap AlarmHandler 14
PROG=$1
SetTimer 15
$PROG > /dev/null 2>&1 &
CHPROCIDS=$!
wait $!
UnsetTimer
echo "ALL Done."
exit 0


执行上述脚本可能有两种情况:一是指定要执行的程序很快结束,另一种是指定要执行的程序耗时较长,超过15秒钟,由定时器结束.如:

1) #./aaa "date" #date的执行需时很短,结果显示"ALL Done."

2) #./aaa "find / -name jjjj " #find的执行需时很长,在经过15秒后,程序结束,结果显示"Sub-processes killed."

抱歉!评论已关闭.