當我們在win上遇到一個沒有辦法正常關閉的軟件的時候,我們可以打開任務管理器,強制結束這個進程,同理。在linux上也有類似的功能,比如從終端敲入ctrl+c組合鍵來產生一個信號,當運行中的進程捕捉到這個信號后就會做出反應。 我們在win上可以使用快捷鍵進入任務管理器,那么linux中會在哪些情況下會產生信號呢?
1042070qdq6x0khacs0x45.jpg (5.45 KB, 下載次數: 50)
下載附件
2019-9-11 02:19 上傳
那么linux有多少種信號呢,我們可以通過命令kill命令來查看,如下圖:
104207yh5ahnlxshnprl1p.jpg (20.29 KB, 下載次數: 46)
下載附件
2019-9-11 02:19 上傳
通過上面的截圖我們可以發現,一共有64種信號,每個信號都是以sig開頭,信號的名稱是在signal.h中定義的。 本次文章主要涉及到三個內容,一,信號的發送,二,信號的接收,三,信號的處理。 第一部分:信號的發送 我們通過一個小例子來認識下信號的發送:
104207al95ztdzaeyd3nzx.jpg (10.81 KB, 下載次數: 50)
下載附件
2019-9-11 02:19 上傳
這個程序主要用到了kill函數, 因為用戶空間是不具備發送信號的能力的,只有內核才可以發信號,內核有那么多信號,我們要發哪個信號,并且發給誰呢?那么我們就需要先告訴內核進程PID,信號ID是多少,kill函數就幫我們解決了這個問題,函數原型:
104207h7ofzhf4555rk7kd.jpg (22.06 KB, 下載次數: 49)
下載附件
2019-9-11 02:19 上傳
所以在上面的例子中,我們就可以通過kill函數來向內核發送一次產生信號的請求。 我們再來看一個和kill函數很像的函數,raise函數 ,與kill函數不同的是,它沒有第一個參數,他不知道要發信號給誰,所以他只能發信號給自己。我們來看下這個例子:
10420706r46odgr6rp69a6.jpg (10.55 KB, 下載次數: 60)
下載附件
2019-9-11 02:19 上傳
編譯并運行:
1042070ht26ftb4o5ouobi.jpg (6.56 KB, 下載次數: 46)
下載附件
2019-9-11 02:19 上傳
函數原型:
1042073tab0c2ozweitoia.jpg (9.53 KB, 下載次數: 48)
下載附件
2019-9-11 02:19 上傳
從kill和raise的函數原型上看,raise沒有pid這個參數,所以raise是可以通過kill來實現的。等價于:
1042075mlivm8zurp7mrrv.jpg (3.63 KB, 下載次數: 43)
下載附件
2019-9-11 02:19 上傳
還有一個需要了解的函數alarm函數,與raise函數不同的是,他只能發alarm信號,并且可以定時發送信號,而raise是立刻讓內核發信號。所以這個函數的參數沒有pid號,也沒有信號ID,只有一個延遲的秒數。需要注意的是,一個進程只能有一個alarm時間,函數原型:
1042082900lz9pzqthzb0p.jpg (6.94 KB, 下載次數: 43)
下載附件
2019-9-11 02:19 上傳
第一部分信號發送總結: 用戶空間不能發送信號,是通過系統調用函數告訴內核發什么信號,發給誰,讓內核來發送的,只有內核才可以發信號。可以使用kill,raise,alarm函數來讓內核發送信號。 第二部分信號接收 我們可以使用pause來接收信號,pause函數使該進程暫停讓出CPU。我們來看下下面這個例子,當我們在鍵盤上按下ctrl+c的時候,程序收到SIGINT信號會被喚醒,然后執行fun函數,處理完之后再返回繼續運行該程序,不按則只打印process start(進入睡眠狀態)。
104208im43k9mxi3zbq4dx.jpg (6.38 KB, 下載次數: 53)
下載附件
2019-9-11 02:19 上傳
函數原型:
104208nrp9pyzrbpx6rxup.jpg (5.45 KB, 下載次數: 38)
下載附件
2019-9-11 02:19 上傳
第三部分信號處理 信號的處理有三種方式,分別為:1,忽略,就是收到信號后,什么也不做,不處理。2,按照默認的方式處理。3,捕獲并處理,捕獲到信號后,執行我們自己想執行的代碼。我們先來看下signal函數:
104208iayngctc4dyaquib.jpg (8.67 KB, 下載次數: 42)
下載附件
2019-9-11 02:19 上傳
第一種處理方式,忽略: 我們來看下這個例子。
104208anzx0jdj6dd1a2y2.jpg (6.68 KB, 下載次數: 42)
下載附件
2019-9-11 02:19 上傳
編譯并運行,因為我們使用的參數為SIG_IGN,所以我們按下ctrl+c的時候并不能中斷程序運行.
104208qtxtid8hto4cx0cn.jpg (9.04 KB, 下載次數: 47)
下載附件
2019-9-11 02:19 上傳
第二種處理方式,按照默認的方式處理,我們把上個例子中的參數改成SIG_DFL,如下:
1042081smvik45c1ci7ksv.jpg (7.15 KB, 下載次數: 48)
下載附件
2019-9-11 02:19 上傳
編譯并運行,當我們按下ctrl+c的時候,會中斷我們程序。
104208j8jptpl8hg9xl7l7.jpg (8.53 KB, 下載次數: 46)
下載附件
2019-9-11 02:19 上傳
第三種處理方式,執行我們自己的代碼:
104208xpa8hq6appiiq7bw.jpg (7.8 KB, 下載次數: 48)
下載附件
2019-9-11 02:19 上傳
當我們按下ctrl+c的時候,會進去到fun函數。
104208gw55uuswulo0utgu.jpg (7.73 KB, 下載次數: 49)
下載附件
2019-9-11 02:19 上傳
三個部分總結: 1,我們可以使用kill命令來查看有多少個信號,在這么多信號中,我們要格外記住以下幾個:
104208zgys9wgbgts9sjbn.jpg (10.86 KB, 下載次數: 40)
下載附件
2019-9-11 02:19 上傳
2,用戶空間不能發送信號,信號的產生來自內核,讓內核產生信號的方式有:通過鍵盤輸入ctrl+c等。當程序運行出錯時,內核會給進程發送一個信號,如浮點溢出等,還有就是一個程序可以通過調用函數來給另外一個進程發信號,如kill。 3,進程收到信號后,可以忽略,或者按照默認的方式處理,或者按照自己的處理函數來處理,signale是永久注冊的,每次都有效,如果不想的話這樣可以使用sigaction。 |