一. 什么是守护进程
守护进程(daemon)是一类在后台运行的特殊进程,用于执行特定的系统任务。很多守护进程在系统引导的时候启动,并且一直运行直到系统关闭。另一些只在需要的时候才启动,完成任务后就自动结束。
在linux系统中,我们会发现在系统启动的时候有很多的进程就已经开始跑了,也称为服务,这也是我们所说的守护进程。
守护进程是脱离于终端并且在后台运行的进程,脱离终端是为了避免在执行的过程中的信息在终端上显示,并且进程也不会被任何终端所产生的终端信息所打断。
守护进程一般的生命周期是系统启动到系统停止运行,也可以通过杀死进程的方式来结束进程的生命周期。linux系统中有很多的守护进程,最典型的就是我们经常看到的服务进程。当然,我们也经常会利用守护进程来完成很多的系统或者自动化任务。
二. 为什么要设置守护进程
三. 设置守护进程的步骤
创建守护进程的步骤:
(1)创建子进程,父进程退出
因为守护进程是脱离终端控制的,所以要造成一种在终端里已经运行完的假象,把所有的工作都放在子进程中去完成。我们知道,父进程退出后,子进程其实就是变成了孤儿进程,孤儿进程一般是由1号进程收养,也就是我们所谓的init进程,也就是说原来的子进程变成了init的子进程。
(2)创建会话
首先看一下进程组的概念:
进程组 :
1.每个进程也属于一个进程组
2.每个进程主都有一个进程组号,该号等于该进程组组长的PID号 .
3.一个进程只能为它自己或子进程设置进程组ID号
会话期:会话期(session)是一个或多个进程组的集合。
setsid()函数可以建立一个对话期:
如果,调用setsid的进程不是一个进程组的组长,此函数创建一个新的会话期。
(1)此进程变成该对话期的首进程
(2)此进程变成一个新进程组的组长进程。
(3)此进程没有控制终端,如果在调用setsid前,该进程有控制终端,那么与该终端的联系被解除。 如果该进程是一个进程组的组长,此函数返回错误。
(4)为了保证这一点,我们先调用fork()然后exit(),此时只有子进程在运行
其实就是起到了进程脱离原会话的控制,摆脱了原进程组的控制,摆脱了原控制终端的控制。
那么为什么要调用setsid函数呢?主要是因为前面创建子进程的时候,随人父进程退出了,但是子进 程拷贝了父进程的会话、进程组、控制终端等等,所以需要使用setsid来摆脱控制。
(3)改变当前目录为根目录
在fork子进程的时候,子进程也继承了父进程的工作目录。通常是让‘/’目录作为守护进程的当前目录,可以避免很多麻烦。
(4)设置文件掩码
同一个文件的权限掩码是一样的,我们使用fork创建的子进程会从父进程那里继承一些已经打开的文件,打开的文件可能永远不会被守护进程读写,但是他们一样消耗资源,而且可能导致所在的文件系统无法结束。
用例1
#include <iostream>
#include <unistd.h>
#include <fstream>
#include <fcntl.h>
void daemonize()
{
if(fork() != 0) //将父进程退出
exit(0);
setsid(); //设置新的会话
int fd = 0;
if ((fd = open("/dev/null", O_RDWR, 0)) != -1) { //打开空设备文件
dup2(fd, STDIN_FILENO); //将0重定向到/dev/null
dup2(fd, STDOUT_FILENO); //将1重定向到/dev/null
dup2(fd, STDERR_FILENO); //将2重定向到/dev/null
if(fd > STDERR_FILENO)
close(fd);
}
if ((fd = open("test.log", O_RDWR|O_APPEND|O_CREAT, 0))!= -1) {
dup2(fd, STDOUT_FILENO); // 将2重定向到日志文件
if(fd>STDERR_FILENO)
close(fd);
}
}
int main()
{
daemonize();
while(1) {
}
return 0;
}
用例2
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#define MAXFILE 65535
int main()
{
int i,fd;
pid_t pid;
char *buf = "hello,dameon";
int len = strlen(buf);
//第一步,创建子进程
pid = fork();
if (pid < 0) {
printf("error\n");
exit(1);//异常退出
}
else if (pid > 0) {
exit(0);
}
//第二步,创建会话
setsid();
//第三步,修改根目录
chdir("/");
//第四步,设置掩码
umask(0);
for (i=0;i<MAXFILE;i++) {
close(i);
}
while(1) {
fd = open("/mnt/hgfs/H/C/dameon.log",O_CREAT|O_WRONLY,0600);
if(fd < 0) {
perror("open");
exit(1);
}
write(fd,buf,len+1);
close(fd);
}
return 0;
}
参考:
[1] https://zhuanlan.zhihu.com/p/56840430