C++編寫LINUX守護進程的實現(xiàn)代碼
1、什么是守護進程
守護進程是運行在后臺的一種特殊進程,它獨立于控制終端并且周期性地執(zhí)行某種任務(wù)或循環(huán)等待處理某些事件的發(fā)生;
守護進程一般在系統(tǒng)啟動時開始運行,除非強行終止,否則直到系統(tǒng)關(guān)機才隨之一起停止運行;
守護進程一般都以root用戶權(quán)限運行,因為要使用某些特殊的端口或者資源;
守護進程的父進程一般都是init進程,因為它真正的父進程在fork出守護進程后就直接退出了,所以守護進程都是孤兒進程,由init接管;
2、有哪些常見的守護進程
日志服務(wù)進程 syslogd
數(shù)據(jù)庫守護進程 mysqld
3、創(chuàng)建守護進程的步驟
(1) fork()創(chuàng)建子進程,父進程exit()退出
這是創(chuàng)建守護進程的第一步。由于守護進程是脫離控制終端的,因此,完成第一步后就會在Shell終端里造成程序已經(jīng)運行完畢的假象。之后的所有工作都在子進程中完成,而用戶在Shell終端里則可以執(zhí)行其他命令,從而在形式上做到了與控制終端的脫離,在后臺工作。
(2) 在子進程中調(diào)用 setsid() 函數(shù)創(chuàng)建新的會話
在調(diào)用了fork()函數(shù)后,子進程全盤拷貝了父進程的會話期、進程組、控制終端等,雖然父進程退出了,但會話期、進程組、控制終端等并沒有改變,因此,這還不是真正意義上的獨立開來,而 setsid() 函數(shù)能夠使進程完全獨立出來。
(3) 再次 fork() 一個孫進程并讓子進程退出
為什么要再次fork呢,假定有這樣一種情況,之前的父進程fork出子進程以后還有別的事情要做,在做事情的過程中因為某種原因阻塞了,而此時的子進程因為某些非正常原因要退出的話,就會形成僵尸進程,所以由子進程fork出一個孫進程以后立即退出,孫進程作為守護進程會被init接管,此時無論父進程想做什么都隨它了。
(4) 在孫進程中調(diào)用 chdir() 函數(shù),讓根目錄 ”/” 成為孫進程的工作目錄
這一步也是必要的步驟,使用fork創(chuàng)建的子進程繼承了父進程的當(dāng)前工作目錄。由于在進程運行中,當(dāng)前目錄所在的文件系統(tǒng)(如“/mnt/usb”)是不能卸載的,這對以后的使用會造成諸多的麻煩(比如系統(tǒng)由于某種原因要進入單用戶模式)。因此,通常的做法是讓"/"作為守護進程的當(dāng)前工作目錄,這樣就可以避免上述的問題,當(dāng)然,如有特殊需要,也可以把當(dāng)前工作目錄換成其他的路徑,如/tmp,改變工作目錄的常見函數(shù)是chdir。
(5) 在孫進程中調(diào)用 umask() 函數(shù),設(shè)置進程的文件權(quán)限掩碼為0
文件權(quán)限掩碼是指屏蔽掉文件權(quán)限中的對應(yīng)位。比如,有個文件權(quán)限掩碼是050,它就屏蔽了文件組擁有者的可讀與可執(zhí)行權(quán)限。由于使用fork函數(shù)新建的子進程繼承了父進程的文件權(quán)限掩碼,這就給該子進程使用文件帶來了諸多的麻煩。因此,把文件權(quán)限掩碼設(shè)置為0,可以大大增強該守護進程的靈活性。設(shè)置文件權(quán)限掩碼的函數(shù)是umask。在這里,通常的使用方法為umask(0)。
(6) 在孫進程中關(guān)閉任何不需要的文件描述符
同文件權(quán)限碼一樣,用fork函數(shù)新建的子進程會從父進程那里繼承一些已經(jīng)打開了的文件。這些被打開的文件可能永遠不會被守護進程讀寫,但它們一樣消耗系統(tǒng)資源,而且可能導(dǎo)致所在的文件系統(tǒng)無法卸下。
在上面的第2)步之后,守護進程已經(jīng)與所屬的控制終端失去了聯(lián)系。因此從終端輸入的字符不可能達到守護進程,守護進程中用常規(guī)方法(如printf)輸出的字符也不可能在終端上顯示出來。所以,文件描述符為0、1和2 的3個文件(常說的輸入、輸出和報錯)已經(jīng)失去了存在的價值,也應(yīng)被關(guān)閉。
(7) 守護進程退出處理
當(dāng)用戶需要外部停止守護進程運行時,往往會使用 kill 命令停止該守護進程。所以,守護進程中需要編碼來實現(xiàn) kill 發(fā)出的signal信號處理,達到進程的正常退出。
4、守護進程的代碼實現(xiàn)
#include <unistd.h>
#include <signal.h>
#include <stdlib.h>
#include <string.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <time.h>
#include <stdio.h>
static bool flag = true;
void create_daemon();
void handler(int);
int main()
{
time_t t;
int fd;
create_daemon();
struct sigaction act;
act.sa_handler = handler;
sigemptyset(&act.sa_mask);
act.sa_flags = 0;
if(sigaction(SIGQUIT, &act, NULL))
{
printf("sigaction error.\n");
exit(0);
}
while(flag)
{
fd = open("/home/mick/daemon.log",
O_WRONLY | O_CREAT | O_APPEND, 0644);
if(fd == -1)
{
printf("open error\n");
}
t = time(0);
char *buf = asctime(localtime(&t));
write(fd, buf, strlen(buf));
close(fd);
sleep(60);
}
return 0;
}
void handler(int sig)
{
printf("I got a signal %d\nI'm quitting.\n", sig);
flag = false;
}
void create_daemon()
{
pid_t pid;
pid = fork();
if(pid == -1)
{
printf("fork error\n");
exit(1);
}
else if(pid)
{
exit(0);
}
if(-1 == setsid())
{
printf("setsid error\n");
exit(1);
}
pid = fork();
if(pid == -1)
{
printf("fork error\n");
exit(1);
}
else if(pid)
{
exit(0);
}
chdir("/");
int i;
for(i = 0; i < 3; ++i)
{
close(i);
}
umask(0);
return;
}
5、用系統(tǒng)函數(shù)daemon實現(xiàn)
#include <unistd.h>
#include <signal.h>
#include <stdlib.h>
#include <string.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <time.h>
#include <stdio.h>
static bool flag = true;
void handler(int);
int main()
{
time_t t;
int fd;
if(-1 == daemon(0, 0))
{
printf("daemon error\n");
exit(1);
}
struct sigaction act;
act.sa_handler = handler;
sigemptyset(&act.sa_mask);
act.sa_flags = 0;
if(sigaction(SIGQUIT, &act, NULL))
{
printf("sigaction error.\n");
exit(0);
}
while(flag)
{
fd = open("/home/mick/daemon.log",
O_WRONLY | O_CREAT | O_APPEND, 0644);
if(fd == -1)
{
printf("open error\n");
}
t = time(0);
char *buf = asctime(localtime(&t));
write(fd, buf, strlen(buf));
close(fd);
sleep(60);
}
return 0;
}
void handler(int sig)
{
printf("I got a signal %d\nI'm quitting.\n", sig);
flag = false;
}
以上就是C++實現(xiàn)LINUX守護進程代碼實例的詳細內(nèi)容,更多關(guān)于C++ LINUX守護進程的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
解析在main函數(shù)之前調(diào)用函數(shù)以及對設(shè)計的作用詳解
本篇文章是對在main函數(shù)之前調(diào)用函數(shù)以及對設(shè)計的作用進行了詳細的分析介紹,需要的朋友參考下2013-05-05
C語言可變參數(shù)與函數(shù)參數(shù)的內(nèi)存對齊詳解
這篇文章主要為大家詳細介紹了C語言可變參數(shù)與函數(shù)參數(shù)的內(nèi)存對齊,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助2022-03-03

