一、守护进程定义
守护进程(daemon)是生存期长的一种进程。它们常常在系统引导装入时启动,仅在系统关闭时才终止。因为它们没有控制终端,所以说它们是在后台运行的。
在linux终端中,我们可以用ps -axj 命令来查看系统的守护进程。选项-a显示由其他用户所拥有的进程的状态,-x显示没有控制终端的进程状态,-j显示与作业有关的信息。在接下来的例子中,通过这个命令可以来查看我们的守护进程是否启动了。守护进程最重要的特点是脱离控制终端,这也是决定了我们程序应该如何写。
二、创建守护进程的步骤
我们先来看一个创建守护进程的例子,为了简洁代码,均不做错误判断。
void daemon(void)
{
pid_t pid;
int devnullfd,fd,fdtablesize;
//1.将文件模式创建疲敝字设置为一个已知值(通常是0)。
umask(0);
//2. fork出子进程,然后父进程退出。
pid = fork();
if(pid > 0 )
exit(0);
//3. 调用setsid创建一个新会话。
setsid();
//4. 将当前工作目录更改为根目录。
chdir("/");
//5. 关闭所有不需要的文件描述符。
for(fd = 0, fdtablesize = getdtablesize(); fd < fdtablesize; fd++)
close(fd);
//6. 使标准输入、标准输出、标准错误重定向到/dev/null
devnullfd = open("/dev/null", 0);
dup2(devnullfd, 0);
dup2(devnullfd, 1);
dup2(devnullfd, 2);
//7. 处理SIGCHLD信号
signal(SIGCHLD,SIG_IGN);
return;
}
从以上例子可以看到创建一个守护进程有以下步骤(顺序可以调整):
- 将文件模式创建疲敝字设置为一个已知值(通常是0)。
设置允许当前进程创建文件或者目录最大可操作的权限,比如这里设置为0,它的意思就是0取反再创建文件时权限相与,也就是:(~0) & mode 等于八进制的值0777 & mode了,这样给后面的代码调用函数mkdir给出最大的权限,避免了创建目录或文件的权限不确定性。子进程继承了父进程的文件创建屏蔽字,同样也是给出mkdir最大的权限。 - fork出子进程,然后父进程退出。
父进程终止,子进程运行。虽然子进程继承了父进程的组ID,但是获得了一个新的进程ID,两者不可能相等,这就保证了子进程不是一个进程组的组长。这也是下面调用setsid函数的先决条件。 - 调用setsid创建一个新会话。
若成功,返回进程组ID,若失败,返回-1。
#include<unistd.h>
pid_t setsid(void);
调用setsid创建一个新会话。使调用进程:a. 成为新会话的首进程;b. 成为一个新进程组的组长进程;c. 没有控制终端。到这里,如果创建成功,程序就基本脱离了终端的控制了,这也是我们的目的。 - 将当前工作目录更改为根目录。
从父进程继承过来的当前目录可能是在一个挂载的文件系统中,因为守护进程在引导之前是一直存在的,所以如果守护进程的当前工作目录在一个挂载文件系统中,将导致该文件系统不能被卸载。 - 关闭所有不需要的文件描述符。
子进程会从父进程继承文件描述符,因为不再