守護進程

守護進程

來自專欄我有一個程序員朋友

Linux Daemon(守護進程)是運行在後台的一種特殊進程。它獨立於控制終端並且周期性地執行某種任務或等待處理某些發生的事件。它不需要用戶輸入就能運行而且提供某種服務,不是對整個系統就是對某個用戶程序提供服務。Linux系統的大多數伺服器就是通過守護進程實現的。常見的守護進程包括系統日誌進程syslogd、 web伺服器httpd、郵件伺服器sendmail和資料庫伺服器mysqld等。

創建守護進程的的一般步驟:

1、fork()創建子進程,父進程exit()退出

這是創建守護進程的第一步。由於守護進程是脫離控制終端的,因此,完成第一步後就會在Shell終端里造成程序已經運行完畢的假象。之後的所有工作都在子進程中完成,而用戶在Shell終端里則可以執行其他命令,從而在形式上做到了與控制終端的脫離,在後台工作。

2、在子進程中調用 setsid() 函數創建新的會話

在調用了 fork() 函數後,子進程全盤拷貝了父進程的會話期、進程組、控制終端等,雖然父進程退出了,但會話期、進程組、控制終端等並沒有改變,因此,這還不是真正意義上的獨立開來,而 setsid() 函數能夠使進程完全獨立出來。

3、再次 fork() 一個子進程並讓父進程退出。

現在,進程已經成為無終端的會話組長,但它可以重新申請打開一個控制終端,可以通過 fork() 一個子進程,該子進程不是會話首進程,該進程將不能重新打開控制終端。退出父進程。

4、在子進程中調用 chdir() 函數,讓根目錄 」/」 成為子進程的工作目錄

這一步也是必要的步驟。使用fork創建的子進程繼承了父進程的當前工作目錄。由於在進程運行中,當前目錄所在的文件系統(如「/mnt/usb」)是不能卸載的,這對以後的使用會造成諸多的麻煩(比如系統由於某種原因要進入單用戶模式)。因此,通常的做法是讓"/"作為守護進程的當前工作目錄,這樣就可以避免上述的問題,當然,如有特殊需要,也可以把當前工作目錄換成其他的路徑,如/tmp。改變工作目錄的常見函數是chdir。

5、在子進程中調用 umask() 函數,設置進程的文件許可權掩碼為0

文件許可權掩碼是指屏蔽掉文件許可權中的對應位。比如,有個文件許可權掩碼是050,它就屏蔽了文件組擁有者的可讀與可執行許可權。由於使用fork函數新建的子進程繼承了父進程的文件許可權掩碼,這就給該子進程使用文件帶來了諸多的麻煩。因此,把文件許可權掩碼設置為0,可以大大增強該守護進程的靈活性。設置文件許可權掩碼的函數是umask。在這裡,通常的使用方法為umask(0)。

6、在子進程中關閉任何不需要的文件描述符

同文件許可權碼一樣,用fork函數新建的子進程會從父進程那裡繼承一些已經打開了的文件。這些被打開的文件可能永遠不會被守護進程讀寫,但它們一樣消耗系統資源,而且可能導致所在的文件系統無法卸下。在上面的第二步之後,守護進程已經與所屬的控制終端失去了聯繫。因此從終端輸入的字元不可能達到守護進程,守護進程中用常規方法(如printf)輸出的字元也不可能在終端上顯示出來。所以,文件描述符為0、1和2 的3個文件(常說的輸入、輸出和報錯)已經失去了存在的價值,也應被關閉。

7、守護進程退出處理

當用戶需要外部停止守護進程運行時,往往會使用 kill 命令停止該守護進程。所以,守護進程中需要編碼來實現 kill 發出的signal信號處理,達到進程的正常退出。

8、示例代碼

void init_daemon(){ pid_t pid; if ((pid = fork()) != 0) exit(0); setsid(); if ((pid = fork()) != 0) exit(0); chdir(/); umask(0);}int main (int argc, char *argv[]){ init_daemon(); while (1) { } return 0;}


歡迎關注我的公眾號-我有個程序員朋友

weixin.qq.com/r/yD-p8eH (二維碼自動識別)

推薦閱讀:

一個用慣了windows的用戶,怎樣上手linux系統?如果用linux系統,能夠勝任日常辦公需求嗎?
如何說服Linus Torvalds用Go語言完全重寫Kernel 5.0版本?
做linux kernel相關研究的如何創(zhuan)業(qian)?
linux整理
購置筆記本電腦,Ubuntu專用,求品牌型號推薦?

TAG:Linux | 計算機科學 | 編程 |