原文地址:http://koudai.360.cn/u/18497921/article_125515949.html?s=y
參考地址:http://www.cnblogs.com/guosixu/archive/2008/07/11/1240821.html
一. trap捕捉到信號之后,可以有三種反應方式:
(1)執(zhí)行一段程序來處理這一信號
(2)接受信號的默認操作
(3)忽視這一信號
二. trap對上面三種方式提供了三種基本形式:
- 第一種形式的trap命令在shell接收到signal list清單中數(shù)值相同的信號時,將執(zhí)行雙引號中的命令串。
- trap 'commands' signal-list
- trap "commands" signal-list
-
- 為了恢復信號的默認操作,使用第二種形式的trap命令:
- trap signal-list
-
- 第三種形式的trap命令允許忽視信號
- trap " " signal-list
第一種形式的trap命令在shell接收到signal list清單中數(shù)值相同的信號時,將執(zhí)行雙引號中的命令串。
trap 'commands' signal-list
trap "commands" signal-list
為了恢復信號的默認操作,使用第二種形式的trap命令:
trap signal-list
第三種形式的trap命令允許忽視信號
trap " " signal-list
注意:
(1) 對信號11(段違例)不能捕捉,因為shell本身需要捕捉該信號去進行內存的轉儲。
(2) 在trap中可以定義對信號0的處理(實際上沒有這個信號), shell程序在其終止(如執(zhí)行exit語句)時發(fā)出該信號。
(3) 在捕捉到signal-list中指定的信號并執(zhí)行完相應的命令之后, 如果這些命令沒有將shell程序終止的話,shell程序將繼續(xù)執(zhí)行收到信號時所執(zhí)行的命令后面的命令,這樣將很容易導致shell程序無法終止。
另外,在trap語句中,單引號和雙引號是不同的,當shell程序第一次碰到trap語句時,將把commands中的命令掃描一遍。此時若commands是用單引號括起來的話,那么shell不會對commands中的變量和命令進行替換, 否則commands中的變量和命令將用當時具體的值來替換。
在有些情況下,我們不希望自己的shell腳本在運行時刻被中斷,比如說我們寫得shell腳本設為某一用戶的默認shell,使這一用戶進入系統(tǒng)后只能作某一項工作,如數(shù)據(jù)庫備份, 我們可不希望用戶使用ctrl+C之類便進入到shell狀態(tài),做我們不希望做的事情。這便用到了信號處理。
kill -l可以列出系統(tǒng)的信號名稱,如下:
- -bash-3.00$ kill -l
- 1) SIGHUP 2) SIGINT 3) SIGQUIT 4) SIGILL
- 5) SIGTRAP 6) SIGABRT 7) SIGEMT 8) SIGFPE
- 9) SIGKILL 10) SIGBUS 11) SIGSEGV 12) SIGSYS
- 13) SIGPIPE 14) SIGALRM 15) SIGTERM 16) SIGUSR1
- 17) SIGUSR2 18) SIGCHLD 19) SIGPWR 20) SIGWINCH
- 21) SIGURG 22) SIGIO 23) SIGSTOP 24) SIGTSTP
- 25) SIGCONT 26) SIGTTIN 27) SIGTTOU 28) SIGVTALRM
- 29) SIGPROF 30) SIGXCPU 31) SIGXFSZ 32) SIGWAITING
- 33) SIGLWP 34) SIGFREEZE 35) SIGTHAW 36) SIGCANCEL
- 37) SIGLOST 41) SIGRTMIN 42) SIGRTMIN+1 43) SIGRTMIN+2
- 44) SIGRTMIN+3 45) SIGRTMAX-3 46) SIGRTMAX-2 47) SIGRTMAX-1
- 48) SIGRTMAX
-bash-3.00$ kill -l
1) SIGHUP 2) SIGINT 3) SIGQUIT 4) SIGILL
5) SIGTRAP 6) SIGABRT 7) SIGEMT 8) SIGFPE
9) SIGKILL 10) SIGBUS 11) SIGSEGV 12) SIGSYS
13) SIGPIPE 14) SIGALRM 15) SIGTERM 16) SIGUSR1
17) SIGUSR2 18) SIGCHLD 19) SIGPWR 20) SIGWINCH
21) SIGURG 22) SIGIO 23) SIGSTOP 24) SIGTSTP
25) SIGCONT 26) SIGTTIN 27) SIGTTOU 28) SIGVTALRM
29) SIGPROF 30) SIGXCPU 31) SIGXFSZ 32) SIGWAITING
33) SIGLWP 34) SIGFREEZE 35) SIGTHAW 36) SIGCANCEL
37) SIGLOST 41) SIGRTMIN 42) SIGRTMIN+1 43) SIGRTMIN+2
44) SIGRTMIN+3 45) SIGRTMAX-3 46) SIGRTMAX-2 47) SIGRTMAX-1
48) SIGRTMAX
通常我們需要忽略的信號有四個,即:HUP, INT, QUIT, TSTP,也就是信號1, 2, 3, 24使用這樣的語句可以使這些中斷信號被忽略:
trap "" 1 2 3 24 或 trap "" HUP INT QUIT TSTP
用 trap :1 2 3 24 或 trap HUP INT QUIT TSTP使其回復默認值。
用stty -a可以列出中斷信號與鍵盤的對應,分別執(zhí)行上面的命令后,運行
tail -f /etc/passwd, 然后嘗試用鍵盤中斷,試試兩種情況(默認和忽略)下有何不同。
- -bash-3.00$ stty -a
- speed 9600 baud;
- rows = 24; columns = 82; ypixels = 0; xpixels = 0;
- csdata ?
- eucw 1:0:0:0, scrw 1:0:0:0
- intr = ^c; quit = ^\; erase = ^?; kill = ^u;
- eof = ^d; eol = <undef>; eol2 = <undef>; swtch = <undef>;
- start = ^q; stop = ^s; susp = ^z; dsusp = ^y;
- rprnt = ^r; flush = ^o; werase = ^w; lnext = ^v;
- -parenb -parodd cs8 -cstopb -hupcl cread -clocal -loblk -crtscts -crtsxoff -parext
-
- -ignbrk brkint -ignpar -parmrk -inpck -istrip -inlcr -igncr icrnl -iuclc
- ixon -ixany -ixoff imaxbel
- isig icanon -xcase echo echoe echok -echonl -noflsh
- -tostop echoctl -echoprt echoke -defecho -flusho -pendin iexten
- opost -olcuc onlcr -ocrnl -onocr -onlret -ofill -ofdel tab3
-bash-3.00$ stty -a
speed 9600 baud;
rows = 24; columns = 82; ypixels = 0; xpixels = 0;
csdata ?
eucw 1:0:0:0, scrw 1:0:0:0
intr = ^c; quit = ^\; erase = ^?; kill = ^u;
eof = ^d; eol = <undef>; eol2 = <undef>; swtch = <undef>;
start = ^q; stop = ^s; susp = ^z; dsusp = ^y;
rprnt = ^r; flush = ^o; werase = ^w; lnext = ^v;
-parenb -parodd cs8 -cstopb -hupcl cread -clocal -loblk -crtscts -crtsxoff -parext
-ignbrk brkint -ignpar -parmrk -inpck -istrip -inlcr -igncr icrnl -iuclc
ixon -ixany -ixoff imaxbel
isig icanon -xcase echo echoe echok -echonl -noflsh
-tostop echoctl -echoprt echoke -defecho -flusho -pendin iexten
opost -olcuc onlcr -ocrnl -onocr -onlret -ofill -ofdel tab3
更方便的是我們可以用在shell中用trap定義我們自己的信號處理程序,就象在c中用signal一樣,
如:trap “echo 'GO Away'” INT
- zj@zj:~/Script/test$ cat reboot.sh
- #/bin/bash
- i=0;
- while(( 1 ))
- do
- echo "hello $i"
- sleep 1
- (( i++ ))
- trap "bash $0 && kill $$" QUIT
- done
- echo haha
- zj@zj:~/Script/test$ ./reboot.sh
- hello 0
- hello 1
- hello 2
- hello 3
- Quit
- hello 0
- hello 1
- hello 2
- Quit
- hello 0
zj@zj:~/Script/test$ cat reboot.sh
#/bin/bash
i=0;
while(( 1 ))
do
echo "hello $i"
sleep 1
(( i++ ))
trap "bash $0 && kill $$" QUIT
done
echo haha
zj@zj:~/Script/test$ ./reboot.sh
hello 0
hello 1
hello 2
hello 3
Quit
hello 0
hello 1
hello 2
Quit
hello 0
這樣^\就可以reboot程序了^_^不知道linux重啟是不是trap到某個信號了就這樣...當然是類似,這個腳本太east了.有時間加機會了看看^_^
附:
Linux信號簡介
1) SIGHUP 本信號在用戶終端連接(正?;蚍钦?結束時發(fā)出, 通常是在終端的控制進程結束時, 通知同一session內的各個作業(yè), 這時它們與控制終端不再關聯(lián).
2) SIGINT 程序終止(interrupt)信號, 在用戶鍵入INTR字符(通常是Ctrl-C)時發(fā)出
3) SIGQUIT 和SIGINT類似, 但由QUIT字符(通常是Ctrl-\)來控制. 進程在因收到SIGQUIT退出時會產生core文件, 在這個意義上類似于一個程序錯誤信號.
4) SIGILL 執(zhí)行了非法指令. 通常是因為可執(zhí)行文件本身出現(xiàn)錯誤, 或者試圖執(zhí)行數(shù)據(jù)段. 堆棧溢出時也有可能產生這個信號.
5) SIGTRAP 由斷點指令或其它trap指令產生. 由debugger使用.
6) SIGABRT 程序自己發(fā)現(xiàn)錯誤并調用abort時產生.
7) SIGIOT 在PDP-11上由iot指令產生, 在其它機器上和SIGABRT一樣.
8) SIGBUS 非法地址, 包括內存地址對齊(alignment)出錯. eg: 訪問一個四個字長的整數(shù), 但其地址不是4的倍數(shù).
9) SIGFPE 在發(fā)生致命的算術運算錯誤時發(fā)出. 不僅包括浮點運算錯誤, 還包括溢出及除數(shù)為0等其它所有的算術的錯誤.
10) SIGKILL 用來立即結束程序的運行. 本信號不能被阻塞, 處理和忽略.
11) SIGUSR1 留給用戶使用
12) SIGSEGV 試圖訪問未分配給自己的內存, 或試圖往沒有寫權限的內存地址寫數(shù)據(jù).
13) SIGUSR2 留給用戶使用
14) SIGPIPE Broken pipe
15) SIGALRM 時鐘定時信號, 計算的是實際的時間或時鐘時間. alarm函數(shù)使用該信號.
16) SIGTERM 程序結束(terminate)信號, 與SIGKILL不同的是該信號可以被阻塞和處理. 通常用來要求程序自己正常退出. shell命令kill缺省產生這個信號.
17) SIGCHLD 子進程結束時, 父進程會收到這個信號.
18) SIGCONT 讓一個停止(stopped)的進程繼續(xù)執(zhí)行. 本信號不能被阻塞. 可以用一個handler來讓程序在由stopped狀態(tài)變?yōu)槔^續(xù)執(zhí)行時完成特定的工作. 例如, 重新顯示提示符
19) SIGSTOP 停止(stopped)進程的執(zhí)行. 注意它和terminate以及interrupt的區(qū)別: 該進程還未結束, 只是暫停執(zhí)行. 本信號不能被阻塞, 處理或忽略.
20) SIGTSTP 停止進程的運行, 但該信號可以被處理和忽略. 用戶鍵入SUSP字符時(通常是Ctrl-Z)發(fā)出這個信號
21) SIGTTIN 當后臺作業(yè)要從用戶終端讀數(shù)據(jù)時, 該作業(yè)中的所有進程會收到SIGTTIN信號. 缺省時這些進程會停止執(zhí)行.
22) SIGTTOU 類似于SIGTTIN, 但在寫終端(或修改終端模式)時收到.
23) SIGURG 有緊急數(shù)據(jù)或out-of-band數(shù)據(jù)到達socket時產生.
24) SIGXCPU 超過CPU時間資源限制. 這個限制可以由getrlimit/setrlimit來讀取/改變
25) SIGXFSZ 超過文件大小資源限制.
26) SIGVTALRM 虛擬時鐘信號. 類似于SIGALRM, 但是計算的是該進程占用的CPU時間.
27) SIGPROF 類似于SIGALRM/SIGVTALRM, 但包括該進程用的CPU時間以及系統(tǒng)調用的時間.
28) SIGWINCH 窗口大小改變時發(fā)出.
29) SIGIO 文件描述符準備就緒, 可以開始進行輸入/輸出操作.
30) SIGPWR Power failure