目录
一,什么是守护进程
守护进程(Daemon Process),也就是通常说的 Daemon 进程(精灵进程),是 Linux 中的后台服务进程。它是一个生存期较长的进程,通常独立于控制终端并且周期性地执行某种任务或等待处理某些发生的事件。
守护进程是个特殊的孤儿进程,这种进程脱离终端,为什么要脱离终端呢?之所以脱离于终端是为了避免进程被任何终端所产生的信息所打断,其在执行过程中的信息也不在任何终端上显示。由于在 Linux 中,每一个系统与用户进行交流的界面称为终端,每一个从此终端开始运行的进程都会依附于这个终端,这个终端就称为这些进程的控制终端,当控制终端被关闭时,相应的进程都会自动关闭。
二,怎么编写守护进程
1,在后台运行
这是为避免挂起控制终端将守护进程放入后台执行。方法是在进程中调用 fork() 使父进程终 止, 让守护进行在子进程中后台执行。
if(fork()>0)
{
exit(0);
}
2,脱离控制终端、登录会话和进程组
Linux 中的进程与控制终端,登录会话和进程组之间的关系:进程属于一个进程组,进程组号(GID)就是进程组长的进程号(PID)。登录会话可以包含多个进程组。这些进程组共享一个控制终端。这个控制终端通常是创建进程的 shell 登录终端。 控制终端、登录会话和进程组通常是从父进程继承下来的。我们的目的就是要摆脱它们 ,使之不受它们的影响。因此需要调用 setsid() 使子进程成为新的会话组长,示例代码如下:
setsid();
setsid() 调用成功后,进程成为新的会话组长和新的进程组长,并与原来的登录会话和进程组脱离。由于会话过程对控制终端的独占性,进程同时与控制终端脱离
3,禁止进程重新打开控制终端
现在,进程已经成为无终端的会话组长,但它可以重新申请打开一个控制终端。可以通过使进程不再成为会话组长来禁止进程重新打开控制终端,采用的方法是再次创建一个子进程,示例代码如下:
if(fork()>0){
//father
exit(0);
}
//不是必须的,防御性的编程
if(fork() > 0){
//father
exit(0);
}
//其目的是让孙子进程跑,话首也被更改,所以调不动进程到前台
4,关闭打开的文件描述符
进程从创建它的父进程那里继承了打开的文件描述符。如不关闭,将会浪费系统资源,造成进程所在的文件系统无法卸下以及引起无法预料的错误。按如下方法关闭它们:
close(0);
dup2("dev/NULL",1);
dup2("dev/NULL",2);
5,改变当前工作目录
进程活动时,其工作目录所在的文件系统不能卸下。一般需要将工作目录改变到根目录。对于需要转储核心,写运行日志的进程将工作目录改变到特定目录如 /tmp。示例代码如下:
chdir("/");
6,重设文件创建掩模
进程从创建它的父进程那里继承了文件创建掩码。它可能修改守护进程所创建的文件的存取权限。为防止这一点,将文件创建掩模清除。
umask(0);
7,处理 SIGCHLD 信号
对于子进程发出信号,要做出处理,如果不处理,就有可能变成僵尸进程,造成内存泄露,资源浪费。
signal(SIGCHLD, SIG_IGN);
代码总的如下:
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <signal.h>
#include <stdlib.h>
#include <fcntl.h>
void my_daemon(int nochdir, int noclose)
{
umask(0);
if(fork()>0){
//father
exit(0);
}
signal(SIGCHLD, SIG_IGN);
setsid();
//不是必须的,防御性的编程
if(fork() > 0){
//father
exit(0);
}
if(nochdir == 0){
chdir("/"); //可选的选项
}
//孙子进程
//可选的选项
if(noclose == 0){
close(0);
//close(1);
//close(2);
int fd = open("/dev/null", O_RDWR);
if(fd < 0){
return; //??
}
dup2(fd, 1);
dup2(fd, 2);
}
}
int main()
{
my_daemon(0, 0);
while(1){
//todo 某种周期性任务
sleep(1);
}
}