欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Ruby信號(hào)處理詳解

 更新時(shí)間:2022年04月18日 11:42:34   作者:駿馬金龍  
這篇文章主要介紹了Ruby信號(hào)處理詳解,包含了Ruby使用Process.kill發(fā)送信號(hào),Ruby使用trap()設(shè)置信號(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.wait

Ruby使用trap()設(shè)置信號(hào)處理程序

Ruby中使用Kernel.trapSignal.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
=end

trap的第一個(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ù)為字符串IGNORESIG_IGN,表示忽略本次捕獲的信號(hào)
  • 如果第二個(gè)參數(shù)為字符串DEFAULTSIG_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)鏈接

相關(guān)文章

最新評(píng)論