trap 命令詳解

 一. trap捕捉到信號之後,可以有三種反應方式:
  (1)執行一段程序來處理這一信號
  (2)接受信號的默認操作
  (3)忽視這一信號

  二. trap對上面三種方式提供了三種基本形式:
  

Shell代碼 複製代碼
  1. 第一種形式的trap命令在shell接收到signal list清單中數值相同的信號時,將執行雙引號中的命令串。   
  2.   trap 'commands' signal-list   
  3.   trap "commands" signal-list   
  4.   
  5.   爲了恢復信號的默認操作,使用第二種形式的trap命令:   
  6.   trap signal-list   
  7.   
  8.   第三種形式的trap命令允許忽視信號   
  9.   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中的變量和命令將用當時具體的值來替換。


在有些情況下,我們不希望自己的shell腳本在運行時刻被中斷,比如說我們寫得shell腳本設爲某一用戶的默認shell,使這一用戶進入系統後只能作某一項工作,如數據庫備份, 我們可不希望用戶使用ctrl+C之類便進入到shell狀態,做我們不希望做的事情。這便用到了信號處理。

kill -l可以列出系統的信號名稱,如下:

Shell代碼 複製代碼
  1. -bash-3.00$ kill -l   
  2.  1) SIGHUP       2) SIGINT       3) SIGQUIT      4) SIGILL   
  3.  5) SIGTRAP      6) SIGABRT      7) SIGEMT       8) SIGFPE   
  4.  9) SIGKILL     10) SIGBUS      11) SIGSEGV     12) SIGSYS   
  5. 13) SIGPIPE     14) SIGALRM     15) SIGTERM     16) SIGUSR1   
  6. 17) SIGUSR2     18) SIGCHLD     19) SIGPWR      20) SIGWINCH   
  7. 21) SIGURG      22) SIGIO       23) SIGSTOP     24) SIGTSTP   
  8. 25) SIGCONT     26) SIGTTIN     27) SIGTTOU     28) SIGVTALRM   
  9. 29) SIGPROF     30) SIGXCPU     31) SIGXFSZ     32) SIGWAITING   
  10. 33) SIGLWP      34) SIGFREEZE   35) SIGTHAW     36) SIGCANCEL   
  11. 37) SIGLOST     41) SIGRTMIN    42) SIGRTMIN+1  43) SIGRTMIN+2  
  12. 44) SIGRTMIN+3  45) SIGRTMAX-3  46) SIGRTMAX-2  47) SIGRTMAX-1  
  13. 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可以列出中斷信號與鍵盤的對應,分別執行上面的命令後,運行
tail -f /etc/passwd, 然後嘗試用鍵盤中斷,試試兩種情況(默認和忽略)下有何不同。

Shell代碼 複製代碼
  1. -bash-3.00$ stty -a   
  2. speed 9600 baud;   
  3. rows = 24; columns = 82; ypixels = 0; xpixels = 0;   
  4. csdata ?   
  5. eucw 1:0:0:0, scrw 1:0:0:0  
  6. intr = ^c; quit = ^/; erase = ^?; kill = ^u;   
  7. eof = ^d; eol = <undef>; eol2 = <undef>; swtch = <undef>;   
  8. start = ^q; stop = ^s; susp = ^z; dsusp = ^y;   
  9. rprnt = ^r; flush = ^o; werase = ^w; lnext = ^v;   
  10. -parenb -parodd cs8 -cstopb -hupcl cread -clocal -loblk -crtscts -crtsxoff -parext   
  11.   
  12. -ignbrk brkint -ignpar -parmrk -inpck -istrip -inlcr -igncr icrnl -iuclc   
  13. ixon -ixany -ixoff imaxbel   
  14. isig icanon -xcase echo echoe echok -echonl -noflsh   
  15. -tostop echoctl -echoprt echoke -defecho -flusho -pendin iexten   
  16. opost -olcuc onlcr -ocrnl -onocr -onlret -ofill -ofdel tab3  



更方便的是我們可以用在shell中用trap定義我們自己的信號處理程序,就象在c中用signal一樣,

如:trap “echo 'GO Away'” INT

Shell代碼 複製代碼
  1. zj@zj:~/Script/test$ cat reboot.sh   
  2. #/bin/bash   
  3. i=0;   
  4. while(( 1 ))   
  5. do   
  6. echo "hello $i"  
  7. sleep 1  
  8. (( i++ ))   
  9. trap "bash $0 && kill $$" QUIT   
  10. done   
  11. echo haha   
  12. zj@zj:~/Script/test$ ./reboot.sh   
  13. hello 0  
  14. hello 1  
  15. hello 2  
  16. hello 3  
  17. Quit   
  18. hello 0  
  19. hello 1  
  20. hello 2  
  21. Quit   
  22. hello 0  

這樣^/就可以reboot程序了^_^不知道linux重啓是不是trap到某個信號了就這樣...當然是類似,這個腳本太east了.有時間加機會了看看^_^

附:
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) SIGPIPE Broken 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) SIGPWR Power failure

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章