如何正确编写linux守护进程
1、守護(hù)進(jìn)程,也就是通常說的Daemon進(jìn)程,是Linux中的后臺服務(wù)進(jìn)程。它是一個生存期較長的進(jìn)程,通常獨立于控制終端并且周期性地執(zhí)行某種任務(wù)或等待處理某些發(fā)生的事件。如果想讓某個進(jìn)程不因為用戶或終端或其他地變化而受到影響,那么就必須把這個進(jìn)程變成一個守護(hù)進(jìn)程。
2、創(chuàng)建守護(hù)進(jìn)程步驟
1)創(chuàng)建子進(jìn)程,父進(jìn)程退出
??? 之后的所有工作都在子進(jìn)程中完成,而用戶在Shell終端里則可以執(zhí)行其他命令,從而在形式上做到了與控制終端的脫離。
??? 在Linux中父進(jìn)程先于子進(jìn)程退出會造成子進(jìn)程成為孤兒進(jìn)程,而每當(dāng)系統(tǒng)發(fā)現(xiàn)一個孤兒進(jìn)程時,就會自動由1號進(jìn)程(init)收養(yǎng)它,這樣,原先的子進(jìn)程就會變成init進(jìn)程的子進(jìn)程。
2)在子進(jìn)程中創(chuàng)建新會話
??? 進(jìn)程組:是一個或多個進(jìn)程的集合。進(jìn)程組有進(jìn)程組ID來唯一標(biāo)識。除了進(jìn)程號(PID)之外,進(jìn)程組ID也是一個進(jìn)程的必備屬性。每個進(jìn)程組都有一個組長進(jìn)程,其組長進(jìn)程的進(jìn)程號等于進(jìn)程組ID。且該進(jìn)程組ID不會因組長進(jìn)程的退出而受到影響。
??? 會話周期:會話期是一個或多個進(jìn)程組的集合。通常,一個會話開始于用戶登錄,終止于用戶退出,在此期間該用戶運行的所有進(jìn)程都屬于這個會話期。
(1)pid_t setsid(void);
??? setsid() creates a new session if the calling process is not a process group leader. The calling process will be the only process in this new process group and in this new session.
??? setsid函數(shù)用于創(chuàng)建一個新的會話,并擔(dān)任該會話組的組長。調(diào)用setsid有下面的3個作用:
① 讓進(jìn)程擺脫原會話的控制
② 讓進(jìn)程擺脫原進(jìn)程組的控制
③ 讓進(jìn)程擺脫原控制終端的控制
有以下三個結(jié)果:
(a)成為新會話的首進(jìn)程
(b)成為一個新進(jìn)程組的組長進(jìn)程
(c)沒有控制終端。
有些人建議在此時再次調(diào)用fork,并使父進(jìn)程終止。第二個子進(jìn)程作為守護(hù)進(jìn)程繼續(xù)運行。這樣就保證了該守護(hù)進(jìn)程不是會話首進(jìn)程。
??? setsid函數(shù)能夠使進(jìn)程完全獨立出來,從而擺脫其他進(jìn)程的控制。
??? setsid()調(diào)用成功后,進(jìn)程成為新的會話組長和新的進(jìn)程組長,并與原來的登錄會話和進(jìn)程組脫離。由于會話過程對控制終端的獨占性,進(jìn)程同時與控制終端脫離。 子進(jìn)程可以自己組成一個新的進(jìn)程組,即調(diào)用setpgrp()與原進(jìn)程組脫離關(guān)系,產(chǎn)生一個新的進(jìn)程組,進(jìn)程組號與它的進(jìn)程號相同.這樣,父進(jìn)程退出運行后就不會影響子進(jìn)程的當(dāng)前運行.
3)改變當(dāng)前目錄為根目錄
??? 使用fork創(chuàng)建的子進(jìn)程繼承了父進(jìn)程的當(dāng)前工作目錄;進(jìn)程活動時,其工作目錄所在的文件系統(tǒng)不能卸下。通常的做法是讓"/"作為守護(hù)進(jìn)程的當(dāng)前工作目錄,也可以是其他目錄,如/tmp,使用chdir。
4)重設(shè)文件權(quán)限掩碼
文件權(quán)限掩碼是指屏蔽掉文件權(quán)限中的對應(yīng)位。比如,有個文件權(quán)限掩碼是050,它就屏蔽了文件組擁有者的可讀與可執(zhí)行權(quán)限。mask = mask & ~050
通常,把文件權(quán)限掩碼設(shè)置為0,umask(0)。
5)關(guān)閉文件描述符
用fork函數(shù)新建的子進(jìn)程會從父進(jìn)程那里繼承已經(jīng)打開了的文件描述符。這些被打開的文件可能永遠(yuǎn)不會被守護(hù)進(jìn)程讀寫,但它們一樣消耗系統(tǒng)資源,而且可能導(dǎo)致所在的文件系統(tǒng)無法卸下。
在上面的第二步之后,守護(hù)進(jìn)程已經(jīng)與所屬的控制終端失去了聯(lián)系。因此從終端輸入的字符不可能達(dá)到守護(hù)進(jìn)程,守護(hù)進(jìn)程中用常規(guī)方法(如printf)輸出的字符也不可能在終端上顯示出來。所以,文件描述符為0、1和2 的3個文件(常說的輸入、輸出和報錯)已經(jīng)失去了存在的價值,也應(yīng)被關(guān)閉。
for(i=0;i<MAXFILE;i++)
??? close(i);
6)守護(hù)進(jìn)程退出處理
當(dāng)用戶需要外部停止守護(hù)進(jìn)程運行時,往往會使用 kill命令停止該守護(hù)進(jìn)程。所以,守護(hù)進(jìn)程中需要編碼來實現(xiàn)kill發(fā)出的signal信號處理,達(dá)到進(jìn)程的正常退出。
signal(SIGTERM, sigterm_handler);
void sigterm_handler(int arg)
{
_running = 0;
}
7)處理SIGCHLD信號
??? 處理SIGCHLD信號并不是必須的。但對于某些進(jìn)程,特別是服務(wù)器進(jìn)程往往在請求到來時生成子進(jìn)程處理請求。如果父進(jìn)程不等待子進(jìn)程結(jié)束,子進(jìn)程將成為僵尸進(jìn)程(zombie)從而占用系統(tǒng)資源。如果父進(jìn)程等待子進(jìn)程結(jié)束,將增加父進(jìn)程的負(fù)擔(dān),影響服務(wù)器進(jìn)程的并發(fā)性能。在Linux下可以簡單地將 SIGCHLD信號的操作設(shè)為SIG_IGN。
??? signal(SIGCHLD,SIG_IGN);
??? 這樣,內(nèi)核在子進(jìn)程結(jié)束時不會產(chǎn)生僵尸進(jìn)程。
?
創(chuàng)作挑戰(zhàn)賽新人創(chuàng)作獎勵來咯,堅持創(chuàng)作打卡瓜分現(xiàn)金大獎總結(jié)
以上是生活随笔為你收集整理的如何正确编写linux守护进程的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Redis:05---键的基本命令(下)
- 下一篇: VIM使用系列之一——配置VIM下编程和