Ruby信號(hào)處理詳解
Ruby使用Process.kill發(fā)送信號(hào)
Process.kill(signal, pid, ...) → integer
Process.kill發(fā)送指定的信號(hào)給一個(gè)或多個(gè)進(jìn)程或進(jìn)程組:
- 如果目標(biāo)
pid>0,表示發(fā)送信號(hào)給指定PID的進(jìn)程 - 如果目標(biāo)
pid=0,表示發(fā)送信號(hào)給調(diào)用kill的進(jìn)程所在進(jìn)程組的所有進(jìn)程 - 如果目標(biāo)
pid<0,表示按照操作系統(tǒng)的規(guī)則發(fā)送信號(hào)。對(duì)于Linux來(lái)說(shuō):- 如果
pid=-1,表示發(fā)送信號(hào)給除pid=1的init進(jìn)程外的所有進(jìn)程,當(dāng)然,沒(méi)有權(quán)限的進(jìn)程將不受影響 - 如果
pid<-1,表示發(fā)送信號(hào)給-pid所在進(jìn)程組的所有進(jìn)程,例如-3000表示發(fā)送信號(hào)給pid=3000的進(jìn)程所在進(jìn)程組的所有進(jìn)程
- 如果
Process.kill的第一個(gè)參數(shù)是要發(fā)送的信號(hào):
- 信號(hào)可以是字符串格式的信號(hào)名或數(shù)值格式的信號(hào)ID,INT或SIGINT或1都是有效的信號(hào)
- 如果信號(hào)帶有負(fù)號(hào)(如
-2或-INT),表示發(fā)送信號(hào)給進(jìn)程所在進(jìn)程組而非指定的進(jìn)程(Linux不支持帶負(fù)號(hào)的信號(hào)) - 如果信號(hào)為0,表示探測(cè)是否能發(fā)送信號(hào)給目標(biāo)進(jìn)程,可探測(cè)是否能管理目標(biāo)進(jìn)程或者探測(cè)目標(biāo)進(jìn)程是否存活
pid = fork do
sleep 300
end
# ...
Process.kill("HUP", pid)
Process.waitRuby使用trap()設(shè)置信號(hào)處理程序
Ruby中使用Kernel.trap或Signal.trap捕獲信號(hào)并設(shè)置信號(hào)處理程序,這兩個(gè)trap等價(jià)。
可設(shè)置多個(gè)trap來(lái)監(jiān)控多個(gè)信號(hào)。
Signal.trap(0, proc { puts "Terminating: #{$$}" })
Signal.trap("CLD") { puts "Child died" }
fork && Process.wait
=begin
Terminating: 27461
Child died
Terminating: 27460
=endtrap的第一個(gè)參數(shù)是監(jiān)控的信號(hào)名稱,可以是字符串的信號(hào)名稱(如SIGINT),可以是省略SIG前綴的信號(hào)名稱(如INT),可以是信號(hào)對(duì)應(yīng)的數(shù)值(如2)。
Ruby支持一個(gè)特殊的信號(hào)0(對(duì)應(yīng)的字符串信號(hào)名為EXIT或SIGEXIT),表示進(jìn)程退出時(shí)會(huì)觸發(fā)的信號(hào)。
trap的第二個(gè)參數(shù)或語(yǔ)句塊是捕獲到信號(hào)后執(zhí)行的代碼。第二個(gè)參數(shù)有幾種特殊情況:
- 如果第二個(gè)參數(shù)為字符串
IGNORE或SIG_IGN,表示忽略本次捕獲的信號(hào) - 如果第二個(gè)參數(shù)為字符串
DEFAULT或SIG_DFL,表示按照Ruby的默認(rèn)處理規(guī)則來(lái)處理 - 如果第二個(gè)參數(shù)為字符串
EXIT,表示以退出狀態(tài)碼0退出當(dāng)前進(jìn)程 - 如果第二個(gè)參數(shù)為字符串
SYSTEM_DEFAULT,表示按照系統(tǒng)的默認(rèn)信號(hào)處理規(guī)則來(lái)處理,即以退出狀態(tài)碼141退出進(jìn)程
避免信號(hào)覆蓋
使用第三方包的時(shí)候,有時(shí)候不知道這個(gè)包是否定義了某個(gè)信號(hào)的信號(hào)處理程序,或者知道它定義了某信號(hào)信號(hào)處理程序,但自己定義這個(gè)信號(hào)的信號(hào)處理程序時(shí),不想覆蓋第三方包中所定義的處理程序。
這時(shí),應(yīng)該利用好trap的返回值。每一次trap設(shè)置信號(hào)處理程序時(shí),都返回本信號(hào)之前已經(jīng)定義的信號(hào)處理程序(是一個(gè)Proc對(duì)象)。只是需要注意,有些信號(hào)的初始處理程序是一個(gè)字符串值DEFAULT而不是一個(gè)Proc對(duì)象,因此,應(yīng)該進(jìn)行類型判斷:
# 第一次定義INT的信號(hào)處理程序
first_trap = trap('INT') {
first_trap.call if first_trap.is_a? Proc
puts "first_trap"
}
# 第二次定義INT的信號(hào)處理程序
old_trap = trap('INT') {
old_trap.call if old_trap.is_a? Proc# 調(diào)用第一次定義的信號(hào)處理程序
puts "old trap" # 本次trap時(shí)執(zhí)行的邏輯
}
# 定義好之后,old_trap為第一次定義的信號(hào)處理程序
# 之后按下CTRL+C觸發(fā)INT信號(hào)的信號(hào)處理程序多線程信號(hào)注冊(cè)問(wèn)題
如果是在多線程中注冊(cè)信號(hào)處理程序,該信號(hào)處理程序?qū)⒖偸亲?cè)在所在進(jìn)程的main線程中(即使是在其它線程中設(shè)置trap())。
pid = fork do
puts "main Thread: #{Thread.current}"
Thread.new {
puts "new Thread: #{Thread.current}"
trap("TERM", proc { puts "Signal: #{Thread.current}" })
sleep 2
}
sleep 2
end
sleep 1
Process.kill 'SIGTERM', pid
=begin
main Thread: #<Thread:0x00007fffd6ed4c10 run>
new Thread: #<Thread:0x00007fffd714f2b0@a.rb:4 run>
Signal: #<Thread:0x00007fffd6ed4c10 run>
=end子進(jìn)程繼承信號(hào)處理程序
子進(jìn)程會(huì)從父進(jìn)程繼承信號(hào)處理程序。
trap 'TERM', proc { puts "Signal: #{Process.pid}" }
puts "Parent: #{Process.pid}"
pid = fork do
sleep 30
end
puts "Child: #{pid}"
Process.kill 'TERM', pid
=begin
Parent: 2872
Child: 2901
Signal: 2901
=end更多關(guān)于Ruby信號(hào)處理的知識(shí)請(qǐng)查看下面的相關(guān)鏈接
版權(quán)聲明:本站文章來(lái)源標(biāo)注為YINGSOO的內(nèi)容版權(quán)均為本站所有,歡迎引用、轉(zhuǎn)載,請(qǐng)保持原文完整并注明來(lái)源及原文鏈接。禁止復(fù)制或仿造本網(wǎng)站,禁止在非maisonbaluchon.cn所屬的服務(wù)器上建立鏡像,否則將依法追究法律責(zé)任。本站部分內(nèi)容來(lái)源于網(wǎng)友推薦、互聯(lián)網(wǎng)收集整理而來(lái),僅供學(xué)習(xí)參考,不代表本站立場(chǎng),如有內(nèi)容涉嫌侵權(quán),請(qǐng)聯(lián)系alex-e#qq.com處理。
關(guān)注官方微信