進(jìn)程控制開(kāi)發(fā)之:Linux守護(hù)進(jìn)程
(2)setsid()函數(shù)格式。
表7.8列出了setsid()函數(shù)的語(yǔ)法規(guī)范。
表7.8 setsid()函數(shù)語(yǔ)法
所需頭文件 | #includesys/types.h> |
函數(shù)原型 | pid_tsetsid(void) |
函數(shù)返回值 | 成功:該進(jìn)程組ID |
3.改變當(dāng)前目錄為根目錄
這一步也是必要的步驟。使用fork()創(chuàng)建的子進(jìn)程繼承了父進(jìn)程的當(dāng)前工作目錄。由于在進(jìn)程運(yùn)行過(guò)程中,當(dāng)前目錄所在的文件系統(tǒng)(比如“/mnt/usb”等)是不能卸載的,這對(duì)以后的使用會(huì)造成諸多的麻煩(比如系統(tǒng)由于某種原因要進(jìn)入單用戶(hù)模式)。因此,通常的做法是讓“/”作為守護(hù)進(jìn)程的當(dāng)前工作目錄,這樣就可以避免上述的問(wèn)題,當(dāng)然,如有特殊需要,也可以把當(dāng)前工作目錄換成其他的路徑,如/tmp。改變工作目錄的常見(jiàn)函數(shù)是chdir()。
4.重設(shè)文件權(quán)限掩碼
文件權(quán)限掩碼是指屏蔽掉文件權(quán)限中的對(duì)應(yīng)位。比如,有一個(gè)文件權(quán)限掩碼是050,它就屏蔽了文件組擁有者的可讀與可執(zhí)行權(quán)限。由于使用fork()函數(shù)新建的子進(jìn)程繼承了父進(jìn)程的文件權(quán)限掩碼,這就給該子進(jìn)程使用文件帶來(lái)了諸多的麻煩。因此,把文件權(quán)限掩碼設(shè)置為0,可以大大增強(qiáng)該守護(hù)進(jìn)程的靈活性。設(shè)置文件權(quán)限掩碼的函數(shù)是umask()。在這里,通常的使用方法為umask(0)。
5.關(guān)閉文件描述符
同文件權(quán)限掩碼一樣,用fork()函數(shù)新建的子進(jìn)程會(huì)從父進(jìn)程那里繼承一些已經(jīng)打開(kāi)了的文件。這些被打開(kāi)的文件可能永遠(yuǎn)不會(huì)被守護(hù)進(jìn)程讀或?qū)懀鼈円粯酉南到y(tǒng)資源,而且可能導(dǎo)致所在的文件系統(tǒng)無(wú)法被卸載。
在上面的第二步之后,守護(hù)進(jìn)程已經(jīng)與所屬的控制終端失去了聯(lián)系。因此從終端輸入的字符不可能達(dá)到守護(hù)進(jìn)程,守護(hù)進(jìn)程中用常規(guī)方法(如printf())輸出的字符也不可能在終端上顯示出來(lái)。所以,文件描述符為0、1和2的3個(gè)文件(常說(shuō)的輸入、輸出和報(bào)錯(cuò)這3個(gè)文件)已經(jīng)失去了存在的價(jià)值,也應(yīng)被關(guān)閉。通常按如下方式關(guān)閉文件描述符:
for(i=0;iMAXFILE;i++)
{
close(i);
}
這樣,一個(gè)簡(jiǎn)單的守護(hù)進(jìn)程就建立起來(lái)了,創(chuàng)建守護(hù)進(jìn)程的流程圖如圖7.7所示。
創(chuàng)建守護(hù)進(jìn)程流程圖
下面是實(shí)現(xiàn)守護(hù)進(jìn)程的一個(gè)完整實(shí)例,該實(shí)例首先按照以上的創(chuàng)建流程建立了一個(gè)守護(hù)進(jìn)程,然后讓該守護(hù)進(jìn)程每隔10s向日志文件/tmp/daemon.log寫(xiě)入一句話。
/*daemon.c創(chuàng)建守護(hù)進(jìn)程實(shí)例*/
#includestdio.h>
#includestdlib.h>
#includestring.h>
#includefcntl.h>
#includesys/types.h>
#includeunistd.h>
#includesys/wait.h>
intmain()
{
pid_tpid;
inti,fd;
char*buf=ThisisaDaemonn;
pid=fork();/*第一步*/
if(pid0)
{
printf(Errorforkn);
exit(1);
}
elseif(pid>0)
{
exit(0);/*父進(jìn)程推出*/
}
setsid();/*第二步*/
chdir(/);/*第三步*/
umask(0);/*第四步*/
for(i=0;igetdtablesize();i++)/*第五步*/
{
close(i);
}
/*這時(shí)創(chuàng)建完守護(hù)進(jìn)程,以下開(kāi)始正式進(jìn)入守護(hù)進(jìn)程工作*/
while(1)
{
if((fd=open(/tmp/daemon.log,
O_CREAT|O_WRONLY|O_APPEND,0600))0)
{
printf(Openfileerrorn);
exit(1);
}
write(fd,buf,strlen(buf)+1);
close(fd);
sleep(10);
}
exit(0);
}
將該程序下載到開(kāi)發(fā)板上,可以看到該程序每隔10s就會(huì)在對(duì)應(yīng)的文件中輸入相關(guān)內(nèi)容。并且使用ps可以看到該進(jìn)程在后臺(tái)運(yùn)行。如下所示:
$tail-f/tmp/daemon.log
ThisisaDaemon
ThisisaDaemon
ThisisaDaemon
ThisisaDaemon
…
$ps-ef|grepdaemon
76root1272S./daemon
85root1520Sgrepdaemon
linux操作系統(tǒng)文章專(zhuān)題:linux操作系統(tǒng)詳解(linux不再難懂)pid控制相關(guān)文章:pid控制原理
linux相關(guān)文章:linux教程
pid控制器相關(guān)文章:pid控制器原理
評(píng)論