守護進程概述
守護進程,又叫daemon進程(不知怎的,我突然想起來吸血鬼日記中的達蒙了,很好 看的美劇),是Linux中的後台服務進程。他是一個生存期較長的進程,通常獨立於控制終端並且周期性地執行 某種任務或者等待處理某些發生的事件。守護進程常常在系統引導載入時啟動,在系統關閉時終止。Linux有 很多系統哦服務,大多數服務都是通過守護進程實現的。同時,守護進程還能完成許多系統任務,例如,作業 規划進程cronf、列印進程lqd等(這裡的結尾字母 d 就是 daemon的意思)。
在Linux中,每一個系統 與使用者進行交流的介面稱為終端,每一個從此終端開始啟動並執行進程都會依附於這個終端,這個終端稱為這些進 程的控制終端,當控制終端關閉時,相應的進程都會自動關閉。但是守護進程卻能夠突破這種限制,它從被執 行開始運轉,直到接收到某種訊號或者整個系統關閉時才退出。如果想讓某個進程不因為使用者、終端或者其它 的變化而受到影響,那麼就必須把這個進程變成一個守護進程。可見,守護進程是非常重要的。
編寫 守護進程步驟
編寫守護進程遵循一個特定的流程,下面就說一下守護進程的建立步驟。
1、創 建子進程,父進程退出。
這是編寫守護進程的第一步。由於守護進程是脫離控制終端的,因此,完成 第一步後就會在shell終端造成一種程式已經運行完畢的假象,之後的所有工作都在子進程中完成,而使用者在 shell終端則可以執行其他的命令,從而在形式上做到與控制終端的脫離。
但是,父進程建立了子進程 後退出,此時該子進程不就沒有父進程了嗎?守護進程中確實會出現這麼一個有趣的現象:由於父進程已經先 於子進程退出,就會造成子進程沒有父進程,從而變成一個孤兒進程。在Linux中,每當系統發現一個孤兒進 程時,就會自動由1號進程(也就是 init 進程)收養它,這樣原先的子進程就會變成 init 進程的子進程。其 關鍵代碼如下;
2、在子進程中建立新會話
這個步驟是建立守護進程最重要的一步,雖然實現非常簡單,但意義卻非常重大。在這裡使用 的是系統函數 setsid(),在具體介紹 setsid()之前,先瞭解以下兩個概念:進程組和會話期。
進程 組。進程組是一個或多個進程的集合。進程組由進程組ID來唯一標識。除了進程號PID之外,進程組ID也是一 個進程的必備屬性。每隔進程組都有一個組長進程,其組長進程的進程號PID等於進程組ID,且該進程組ID不 會因為組長進程的退出而受到影響。(組長沒了,再找個組員來擔任組長唄)
會話期。交談群組是一個或 多個進程組的集合。通常,一個會話開始於使用者登入,終止於使用者退出,在此期間該使用者啟動並執行所有進程都屬 於這個會話期。進程組和會話期之間的關係如圖1所示:
接下來具體介紹 setsid()的相關內 容。
① setsid()函數的作用。setsid()函數用於建立一個新的交談群組,並讓執行此函數的進程擔任該 交談群組的組長。調用setsid()有以下3個作用:
讓進程擺脫原會話的控制
讓進程擺脫原進程 組的控制
讓進程擺脫原控制終端的控制
那麼,回過頭來想想,在建立守護進程時為什麼要調 用 setsid()函數呢?是這樣的,在建立守護進程的第一步中,調用了fork()函數建立子進程再令父進程退出 。由於在調用 fork()函數時,子進程全盤複製了父進程的會話期、進程組和控制終端等,雖然父進程退出了 ,但原先的會話期、進程組和控制終端等並沒有改變,因此,還不是真正意義上的獨立。而setsid()函數能夠 使進程完全獨立出來,從而脫離所有其他進程的控制。
② setsid函數格式