![]() |
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
Linux守護進程 |
||||||||||||||||||||||||||||||||||||||||||||||||||||||||
1.守護進程概述 守護進程,也就是通常所說的daemon進程,是Linux中的后臺服務進程。它是一個生存期較長的進程,通常獨立于控制終端并且周期性地執行某種任務或等待處理某些發生的事件。守護進程常常在系統引導載入時啟動,在系統關閉時終止。Linux有很多系統服務,大多數服務都是通過守護進程實現的。同時,守護進程還能完成許多系統任務,例如,作業規劃進程crond、打印進程lqd等(這里的結尾字母d就是daemon的意思)。 由于在Linux中,每一個系統與用戶進行交流的界面稱為終端,每一個從此終端開始運行的進程都會依附于這個終端,這個終端稱為這些進程的控制終端,當控制終端被關閉時,相應的進程都會自動關閉。但是守護進程卻能夠突破這種限制,它從被執行開始運轉,直到接收到某種信號或者整個系統關閉時才會退出。如果想讓某個進程不因為用戶、終端或者其他的變化而受到影響,那么就必須把這個進程變成一個守護進程。可見,守護進程是非常重要的。 2.編寫守護進程 編寫守護進程看似復雜,但實際上也是遵循一個特定的流程,只要將此流程掌握了,就能很方便地編寫出自己的守護進程。下面就分4個步驟來講解怎樣創建一個簡單的守護進程。在講解的同時,會配合介紹與創建守護進程相關的幾個系統函數,希望讀者能很好地掌握。 (1)創建子進程,父進程退出。這是編寫守護進程的第一步。由于守護進程是脫離控制終端的,因此,完成第一步后就會在shell終端造成一種程序已經運行完畢的假象,之后的所有工作都在子進程中完成,而用戶在shell終端則可以執行其他的命令,從而在形式上做到與控制終端的脫離。 到這里,有心的讀者可能會問,父進程創建了子進程后退出,此時該子進程不就沒有父進程了嗎?守護進程中確實會出現這么一個有趣的現象:由于父進程已經先于子進程退出,就會造成子進程沒有父進程,從而變成一個孤兒進程。在Linux中,每當系統發現一個孤兒進程時,就會自動由1號進程(也就是init進程)收養它,這樣,原先的子進程就會變成init進程的子進程。其關鍵代碼如下: pid = fork(); (2)在子進程中創建新會話。這個步驟是創建守護進程重要的一步,雖然實現非常簡單,但意義卻非常重大。在這里使用的是系統函數setsid(),在具體介紹setsid()之前,讀者首先要了解兩個概念:進程組和會話期。 ● 進程組。進程組是一個或多個進程的集合。進程組由進程組ID來唯一標識。除了進程號(PID)之外,進程組ID也是一個進程的必備屬性。 每個進程組都有一個組長進程,其組長進程的進程號等于進程組ID,且該進程ID不會因組長進程的退出而受到影響。 ● 會話期。會話組是一個或多個進程組的集合。通常,一個會話開始于用戶登錄,終止于用戶退出,在此期間該用戶運行的所有進程都屬于這個會話期。進程組和會話期之間的關系如圖1所示。
接下來就可以具體介紹setsid()的相關內容。 ① setsid()函數的作用。setsid()函數用于創建一個新的會話組,并擔任該會話組的組長。調用setsid()有以下3個作用: 那么,在創建守護進程時為什么要調用setsid()函數呢?讀者可以回憶一下創建守護進程的第一步,在那里調用了fork()函數來創建子進程再令父進程退出。由于在調用fork()函數時,子進程全盤復制了父進程的會話期、進程組和控制終端等,雖然父進程退出了,但原先的會話期、進程組和控制終端等并沒有改變,因此,還不是真正意義上的獨立。而setsid()函數能夠使進程完全獨立出來,從而脫離所有其他進程的控制。 ② setsid()函數格式。表1列出了setsid()函數的語法要點。 表1 setsid()函數語法要點
(3)改變當前目錄為根目錄。這一步也是必要的步驟。使用fork()創建的子進程繼承了父進程的當前工作目錄。由于在進程運行過程中,當前目錄所在的文件系統(如“/mnt/usb”等)是不能卸載的,這對以后的使用會造成諸多的麻煩(如系統由于某種原因要進入單用戶模式)。因此,通常的做法是讓“/”作為守護進程的當前工作目錄,這樣就可以避免上述問題。當然,如有特殊需要,也可以把當前工作目錄換成其他的路徑,如/tmp。改變工作目錄的常見函數是chdir()。 (4)重設文件權限掩碼。文件權限掩碼是指屏蔽掉文件權限中的對應位。例如,有一個文件權限掩碼是050,它就屏蔽了文件組擁有者的可讀與可執行權限。由于使用fork()函數新建的子進程繼承了父進程的文件權限掩碼,這就給該子進程使用文件帶來了諸多的麻煩。因此,把文件權限掩碼設置為0,可以大大增強該守護進程的靈活性。設置文件權限掩碼的函數是umask()。在這里,通常的使用方法為umask(0)。 (5)關閉文件描述符。同文件權限掩碼一樣,用fork()函數新建的子進程會從父進程那里繼承一些已經打開的文件。這些被打開的文件可能永遠不會被守護進程讀或寫,但它們一樣消耗系統資源,而且可能導致所在的文件系統無法被卸載。 在上面的第(2)步之后,守護進程已經與所屬的控制終端失去了聯系,因此,從終端輸入的字符不可能達到守護進程,守護進程中用常規方法(如printf())輸出的字符也不可能在終端上顯示出來。所以,文件描述符為0、1和2的3個文件(常說的輸入、輸出和報錯這3個文件)已經失去了存在的價值,也應被關閉。通常按如下方式關閉文件描述符: for(i = 0; i < MAXFILE; i++) 這樣,一個簡單的守護進程就建立起來了。創建守護進程的流程圖如圖2所示。
下面是實現守護進程的一個完整實例,該實例首先按照以上的創建流程建立了一個守護進程,然后讓該守護進程每隔10s向日志文件/tmp/daemon.log寫入一句話。 /* daemon.c 創建守護進程實例 */ 將該程序下載到開發板上,可以看到該程序每隔10s就會在對應的文件中輸入相關內容,并且使用ps可以看到該進程在后臺運行,結果如下: $ tail -f /tmp/daemon.log 3.守護進程的出錯處理 讀者在前面編寫守護進程的具體調試過程中會發現,由于守護進程完全脫離了控制終端,因此,不能像其他普通進程一樣將錯誤信息輸出到控制終端來通知程序員,即使使用gdb也無法正常調試。那么,守護進程的進程要如何調試呢?一種通用的辦法是使用syslog服務,將程序中的出錯信息輸入到系統日志文件中(如“/var/log/messages”),從而可以直觀地看到程序的問題所在(“/var/log/message”系統日志文件只能由擁有root權限的超級用戶查看。在不同Linux發行版本中,系統日志文件路徑全名可能有所不同,例如,可能是“/var/log/syslog”)。 syslog是Linux中的系統日志管理服務,通過守護進程syslogd來維護。該守護進程在啟動時會讀一個配置文件“/etc/syslog.conf”,該文件決定了不同種類的消息會發送到何處。例如,緊急消息可被送到系統管理員并在控制臺上顯示,而警告消息則可被記錄到一個文件中。 該機制提供了3個syslog相關函數,分別為openlog()、syslog()和closelog(),下面就分別介紹這3個函數。 1)syslog相關函數說明 通常,openlog()函數用于打開系統日志服務的一個鏈接;syslog()函數用于向日志文件中寫入消息,在這里可以規定消息的優先級、消息輸出格式等;closelog()函數用于關閉系統日志服務的鏈接。 2)syslog相關函數格式 表2列出了openlog()函數的語法要點。 表2 openlog()函數語法要點
表3列出了syslog()函數的語法要點。 表3 syslog()函數語法要點
表3.4列出了closelog()函數的語法要點。 表3.4 closelog()函數語法要點
3)使用實例 這里將上一個示例程序用syslog服務進行重寫,其中有區別的地方用加粗的字體表示,源代碼如下: /* syslog_daemon.c利用syslog服務的守護進程實例 */ 讀者可以嘗試用普通用戶的身份執行此程序。由于這里的open()函數必須具有root權限,因此,syslog會將錯誤信息寫入到系統日志文件(如“/var/log/messages”)中,結果如下: Jan 30 18:20:08 localhost daemon_syslog[612]: open 本文選自華清遠見嵌入式培訓教材《從實踐中學嵌入式Linux應用程序開發》 熱點鏈接:
1、Linux下多進程編程之exec函數語法及使用實例 |