概念
Linux的守护进程又被称为Daemon进程,为Linux的后台进程(独立于终端)。一般来说,我们在程序中创建的进程会随着终端的关闭而退出。那么假如我们需要一个在Linux后台不断运行的进程该如何做呢?这个时候就需要我们的守护进程了。守护进程通常周期性地执行着某种任务或者等待着处理某些事件。其生命周期较长,通常在系统启动的时候开始执行,在系统关闭时终止,在Linux中的许多系统服务都是通过守护进程来实现的。一般的网络服务也以守护进程的形式来实现。那么,如何创建一个守护进程呢?
创建守护进程的步骤
- 创建一个子进程,并让其成为孤儿进程
在父进程中创建一个子进程,并让父进程先于子进程退出,此时,子进程便成为了一个孤儿进程。 - 在子进程中创建新会话
由于子进程在创建的时候复制了父进程的会话,进程组和终端控制等,虽然父进程退出了,但是原先的会话,进程组和终端控制等并没有改变,子进程还未完成真正的独立。在Linux系统中可以采用setsid()函数来创建一个新会话,并让进程担任该会话的组长,同时在会话组中创建新的进程组,该进程也是新进程组的组长。从而使得该进程成为新会话组跟新进程组中的唯一的进程,至此该进程完全脱离了终端的控制,运行在后台。
setsid()函数原型如下
#include <unistd.h>
pid_t setsid();
- 改变当前的工作目录
由于采用fork()函数创建的子进程继承了父进程的当前工作目录,而系统通常的做法是让根目录成为守护进程的当前工作目录,所以我们也需要相应的改变子进程的当前工作目录,改变工作目录可以使用chdir()函数。
函数原型如下:
#include <unistd.h>
int chdir(const char* path);
- 重设文件权限掩码
文件权限掩码的作用是屏蔽文件权限中的对应位。由于创建的子进程继承了父进程的文件权限掩码,这给子进程操作文件带来了一定的影响。因此通常需要把文件权限掩码设置为0,这样能够增强守护进程的灵活性。可以使用umask()改变文件权限掩码,参数为要修改的掩码值
该函数原型如下:
#include <sys/types.h>
#include <sys/stat.h>
mode_t umask(mode_t mask);
- 关闭文件描述符
新创建的子进程会从父进程继承一些已经打开的文件描述符,而这些文件描述符可能永远都不会被子进程(守护进程)所访问,但它们却占用了一定的资源。所以,应该关闭,可以使用如下代码进行关闭:
int num = getdtablesize(); //获取当前进程文件描述符表的大小
for(int i = 0; i < num; ++i)
{
close(i);
}
Demo
// this is daemon process example
//
#include <iostream>
#include <cstring>
using namespace std;
#include <sys/types.h>
#include <unistd.h>
#include <sys/wait.h>
#include <sys/stat.h>
#include <stdlib.h>
#include <fcntl.h>
int main(int argc, const char** argv)
{
pid_t pid;
int i, fd;
char* buf = "this is a Daemon\n";
pid = fork();
if(pid < 0)
{
cout << "fork error" << endl;
return -1;
}
else if(pid > 0)
{
// father process
exit(0);
}
else
{
// child process
setsid();
chdir("/tmp");
umask(0);
for(int i = 0; i < getdtablesize(); ++i)
{
close(i); //close the file description flag
}
if(fd = open("daemon.log", O_CREAT|O_WRONLY|O_TRUNC, 0600) < 0)
{
cout << "open error" << endl;
return -1;
}
while(1)
{
write(fd, buf, strlen(buf));
sleep(3);
}
close(fd);
}
return 0;
}
该段程序创建了一个守护进程,并执行每三秒向daemon.log文件输入字符串的任务。
在终端输入ps axj | grep ./daemon
可以看到
可以看到其进程ID,组ID和会话组ID保持一致,说明当前守护进程为组长,“?”代表其为后台进程。