linux 守護進程編寫
守護進程(Daemon)是運行在后臺的一種特殊進程。它獨立于控制終端并且周期性地執(zhí)行某種任務(wù)或等待
本文引用地址:http://m.butianyuan.cn/article/201609/304204.htm處理某些發(fā)生的事件。守護進程是一種很有用的進程。
Linux的大多數(shù)服務(wù)器就是用守護進程實現(xiàn)的。比如,Internet服務(wù)器inetd,Web服務(wù)器httpd等。
同時,守護進程完成許多系統(tǒng)任務(wù)。比如,作業(yè)規(guī)劃進程crond,打印進程lpd等。
守護進程的編程本身并不復(fù)雜,復(fù)雜的是各種版本的Unix的實現(xiàn)機制不盡相同,
造成不同 Unix環(huán)境下守護進程的編程規(guī)則并不一致。
需要注意,照搬某些書上的規(guī)則(特別是BSD4.3和低版本的System V)到Linux會出現(xiàn)錯誤的。
下面結(jié)合一些前輩的文檔和自己的例子說說守護進程的編程。
.基本概念
.進程
.每個進程都有一個父進程
.當子進程終止時,父進程會得到通知并能取得子進程的退出狀態(tài)。
.進程組
.每個進程也屬于一個進程組
.每個進程主都有一個進程組號,該號等于該進程組組長的PID號
.一個進程只能為它自己或子進程設(shè)置進程組ID號
.會話期
.對話期(session)是一個或多個進程組的集合。
.setsid()函數(shù)可以建立一個對話期:
如果,調(diào)用setsid的進程不是一個進程組的組長,此函數(shù)創(chuàng)建一個新的會話期。
(1)此進程變成該對話期的首進程
(2)此進程變成一個新進程組的組長進程。
(3)此進程沒有控制終端,如果在調(diào)用setsid前,該進程有控制終端,那么與該終端的聯(lián)系被解除。
如果該進程是一個進程組的組長,此函數(shù)返回錯誤。
(4)為了保證這一點,我們先調(diào)用fork()然后exit(),此時只有子進程在運行,
子進程繼承了父進程的進程組ID,但是進程PID卻是新分配的,所以不可能是新會話的進程組的PID。
從而保證了這一點。
if((pid=fork())>0) //parent
exit(0);
else if(pid==0){ //th1 child
setsid(); //th1是成為會話期組長
if(fork() ==0){ //th2不會是會話期組長(變成孤兒進程組)
...
}
}
一. 守護進程及其特性
(1)守護進程最重要的特性是后臺運行。在這一點上DOS下的常駐內(nèi)存程序TSR與之相似。
(2)其次,守護進程必須與其運行前的環(huán)境隔離開來。這些環(huán)境包括未關(guān)閉的文件描述符,控制終端,
會話和進程組,工作目錄以及文件創(chuàng)建掩模等。這些環(huán)境通常是守護進程從執(zhí)行它的父進程(特別是shell)
中繼承下來的。
(3)最后,守護進程的啟動方式有其特殊之處。它可以在Linux系統(tǒng)啟動時從啟動腳本/etc/rc.d中啟動,
可以由作業(yè)規(guī)劃進程crond啟動,還可以由用戶終端(通常是 shell)執(zhí)行。
總之,除開這些特殊性以外,守護進程與普通進程基本上沒有什么區(qū)別。
因此,編寫守護進程實際上是把一個普通進程按照上述的守護進程的特性改造成為守護進程。
二. 守護進程的編程要點 (來自UEAP)
前面講過,不同Unix環(huán)境下守護進程的編程規(guī)則并不一致。所幸的是守護進程的編程原則其實都一樣,
區(qū)別在于具體的實現(xiàn)細節(jié)不同。這個原則就是要滿足守護進程的特性。
同時,Linux是基于Syetem V的SVR4并遵循Posix標準,實現(xiàn)起來與BSD4相比更方便。編程要點如下;
1. 在后臺運行。
為避免掛起控制終端將Daemon放入后臺執(zhí)行。方法是在進程中調(diào)用fork使父進程終止,
讓Daemon在子進程中后臺執(zhí)行。
if(pid=fork())
exit(0); //是父進程,結(jié)束父進程,子進程繼續(xù)
2. 脫離控制終端,登錄會話和進程組
進程屬于一個進程組,進程組號(GID)就是進程組長的進程號(PID)。登錄會話可以包含多個進程組。
這些進程組共享一個控制終端。這個控制終端通常是創(chuàng)建進程的登錄終端。
控制終端,登錄會話和進程組通常是從父進程繼承下來的。
我們的目的就是要擺脫它們,使之不受它們的影響。
方法是在第1點的基礎(chǔ)上,調(diào)用setsid()使進程成為會話組長:
setsid();
說明:當進程是會話組長時setsid()調(diào)用失敗。但第一點已經(jīng)保證進程不是會話組長。
setsid()調(diào)用成功后,進程成為新的會話組長和新的進程組長,并與原來的登錄會話和進程組脫離。
由于會話過程對控制終端的獨占性,進程同時與控制終端脫離。
3. 禁止進程重新打開控制終端
現(xiàn)在,進程已經(jīng)成為無終端的會話組長。但它可以重新申請打開一個控制終端。
可以通過使進程不再成為會話組長來禁止進程重新打開控制終端:
if(pid=fork())
exit(0); //結(jié)束第一子進程,第二子進程繼續(xù)(第二子進程不再是會話組長)
4. 關(guān)閉打開的文件描述符
進程從創(chuàng)建它的父進程那里繼承了打開的文件描述符。如不關(guān)閉,將會浪費系統(tǒng)資源,
造成進程所在的文件系統(tǒng)無法卸下以及引起無法預(yù)料的錯誤。按如下方法關(guān)閉它們:
for(i=0;i 關(guān)閉打開的文件描述符close(i);>
5. 改變當前工作目錄
進程活動時,其工作目錄所在的文件系統(tǒng)不能卸下。一般需要將工作目錄改變到根目錄。
對于需要轉(zhuǎn)儲核心,寫運行日志的進程將工作目錄改變到特定目錄如 /tmpchdir(/)
6. 重設(shè)文件創(chuàng)建掩模
進程從創(chuàng)建它的父進程那里繼承了文件創(chuàng)建掩模。它可能修改守護進程所創(chuàng)建的文件的存取位。
為防止這一點,將文件創(chuàng)建掩模清除:umask(0);
7. 處理SIGCHLD信號
處理SIGCHLD信號并不是必須的。
但對于某些進程,特別是服務(wù)器進程往往在請求到來時生成子進程處理請求。
如果父進程不等待子進程結(jié)束,子進程將成為僵尸進程(zombie)從而占用系統(tǒng)資源。
如果父進程等待子進程結(jié)束,將增加父進程的負擔,影響服務(wù)器進程的并發(fā)性能。
在Linux下可以簡單地將 SIGCHLD信號的操作設(shè)為SIG_IGN。
signal(SIGCHLD,SIG_IGN);
這樣,內(nèi)核在子進程結(jié)束時不會產(chǎn)生僵尸進程。
評論