欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C++編寫(xiě)LINUX守護(hù)進(jìn)程的實(shí)現(xiàn)代碼

 更新時(shí)間:2020年06月01日 14:43:50   作者:晟夏的葉  
這篇文章主要介紹了如何使用C++實(shí)現(xiàn)LINUX守護(hù)進(jìn)程,文中代碼非常詳細(xì),供大家學(xué)習(xí)參考,感興趣的小伙伴可以了解下

1、什么是守護(hù)進(jìn)程

守護(hù)進(jìn)程是運(yùn)行在后臺(tái)的一種特殊進(jìn)程,它獨(dú)立于控制終端并且周期性地執(zhí)行某種任務(wù)或循環(huán)等待處理某些事件的發(fā)生;

守護(hù)進(jìn)程一般在系統(tǒng)啟動(dòng)時(shí)開(kāi)始運(yùn)行,除非強(qiáng)行終止,否則直到系統(tǒng)關(guān)機(jī)才隨之一起停止運(yùn)行;

守護(hù)進(jìn)程一般都以root用戶(hù)權(quán)限運(yùn)行,因?yàn)橐褂媚承┨厥獾亩丝诨蛘哔Y源;

守護(hù)進(jìn)程的父進(jìn)程一般都是init進(jìn)程,因?yàn)樗嬲母高M(jìn)程在fork出守護(hù)進(jìn)程后就直接退出了,所以守護(hù)進(jìn)程都是孤兒進(jìn)程,由init接管;

2、有哪些常見(jiàn)的守護(hù)進(jìn)程

日志服務(wù)進(jìn)程 syslogd

數(shù)據(jù)庫(kù)守護(hù)進(jìn)程 mysqld

3、創(chuàng)建守護(hù)進(jìn)程的步驟

(1) fork()創(chuàng)建子進(jìn)程,父進(jìn)程exit()退出

這是創(chuàng)建守護(hù)進(jìn)程的第一步。由于守護(hù)進(jìn)程是脫離控制終端的,因此,完成第一步后就會(huì)在Shell終端里造成程序已經(jīng)運(yùn)行完畢的假象。之后的所有工作都在子進(jìn)程中完成,而用戶(hù)在Shell終端里則可以執(zhí)行其他命令,從而在形式上做到了與控制終端的脫離,在后臺(tái)工作。

(2) 在子進(jìn)程中調(diào)用 setsid() 函數(shù)創(chuàng)建新的會(huì)話

在調(diào)用了fork()函數(shù)后,子進(jìn)程全盤(pán)拷貝了父進(jìn)程的會(huì)話期、進(jìn)程組、控制終端等,雖然父進(jìn)程退出了,但會(huì)話期、進(jìn)程組、控制終端等并沒(méi)有改變,因此,這還不是真正意義上的獨(dú)立開(kāi)來(lái),而 setsid() 函數(shù)能夠使進(jìn)程完全獨(dú)立出來(lái)。

(3) 再次 fork() 一個(gè)孫進(jìn)程并讓子進(jìn)程退出

為什么要再次fork呢,假定有這樣一種情況,之前的父進(jìn)程fork出子進(jìn)程以后還有別的事情要做,在做事情的過(guò)程中因?yàn)槟撤N原因阻塞了,而此時(shí)的子進(jìn)程因?yàn)槟承┓钦T蛞顺龅脑?,就?huì)形成僵尸進(jìn)程,所以由子進(jìn)程fork出一個(gè)孫進(jìn)程以后立即退出,孫進(jìn)程作為守護(hù)進(jìn)程會(huì)被init接管,此時(shí)無(wú)論父進(jìn)程想做什么都隨它了。

(4) 在孫進(jìn)程中調(diào)用 chdir() 函數(shù),讓根目錄 ”/” 成為孫進(jìn)程的工作目錄

這一步也是必要的步驟,使用fork創(chuàng)建的子進(jìn)程繼承了父進(jìn)程的當(dāng)前工作目錄。由于在進(jìn)程運(yùn)行中,當(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。

(5) 在孫進(jìn)程中調(diào)用 umask() 函數(shù),設(shè)置進(jìn)程的文件權(quán)限掩碼為0

文件權(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)。

(6) 在孫進(jìn)程中關(guān)閉任何不需要的文件描述符

同文件權(quán)限碼一樣,用fork函數(shù)新建的子進(jìn)程會(huì)從父進(jìn)程那里繼承一些已經(jīng)打開(kāi)了的文件。這些被打開(kāi)的文件可能永遠(yuǎn)不會(huì)被守護(hù)進(jìn)程讀寫(xiě),但它們一樣消耗系統(tǒng)資源,而且可能導(dǎo)致所在的文件系統(tǒng)無(wú)法卸下。

在上面的第2)步之后,守護(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ò))已經(jīng)失去了存在的價(jià)值,也應(yīng)被關(guān)閉。

(7) 守護(hù)進(jìn)程退出處理

當(dāng)用戶(hù)需要外部停止守護(hù)進(jìn)程運(yùn)行時(shí),往往會(huì)使用 kill 命令停止該守護(hù)進(jìn)程。所以,守護(hù)進(jìn)程中需要編碼來(lái)實(shí)現(xiàn) kill 發(fā)出的signal信號(hào)處理,達(dá)到進(jìn)程的正常退出。

4、守護(hù)進(jìn)程的代碼實(shí)現(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實(shí)現(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++實(shí)現(xiàn)LINUX守護(hù)進(jìn)程代碼實(shí)例的詳細(xì)內(nèi)容,更多關(guān)于C++ LINUX守護(hù)進(jìn)程的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

最新評(píng)論