守护进程(Daemon Process),也就是通常说的 Daemon 进程(精灵进程),是 Linux 中的后台服务进程。它是一个生存期较长的进程,通常独立于控制终端并且周期性地执行某种任务或等待处理某些发生的事件。一般采用以d结尾的名字。
一、进程组
多个进程的集合就是进程组, 这个组中必须有一个组长, 组长就是进程组中的第一个进程,组长以外的都是普通的成员,每个进程组都有一个唯一的组ID,进程组的ID和组长的PID是一样的。
进程组中的成员是可以变动的,如果当前进程组中的成员被转移到了其他组,或者所有进程都推出了,那么进程组也就不存在了。如果进程组中组长死了, 但是当前进程组中有其他进程,这个进程组还是继续存在的。
1.1、获得进程组的组ID
pid_t getpgrp(void);
1.2、获得指定进程所在进程组的组ID
pid_t getpgid(pid_t pid);
pid
指定的进程PID
1.3、创建新的组或移动某个进程
int setpgid(pid_t pid, pid_t pgid);
pid
: 某个进程的进程IDpgid
: 某个进程组的组ID- 如果
pgid
对应的进程组存在,pid对应的进程会移动到这个组中, pid != pgid - 如果
pgid
对应的进程组不存在,会创建一个新的进程组, 因此要求 pid == pgid, 当前进程就是组长了
- 如果
二、会话
会话(session)是由一个或多个进程组组成的,一个会话可以对应一个控制终端, 也可以没有。一个普通的进程可以调用 setsid
函数使自己成为新 session 的领头进程(会长),并且这个 session 领头进程还会被放入到一个新的进程组中。先来看一下setsid()函数的原型:
#include <unistd.h>// 获取某个进程所属的会话ID
pid_t getsid(pid_t pid);// 将某个进程变成会话 =>> 得到一个守护进程
pid_t setsid(void);
- 调用这个函数的进程不能是组长进程, 如果是该函数调用失败,如果保证这个函数能调用成功呢?
- 先fork()创建子进程, 终止父进程, 让子进程调用这个函数
- 如果调用这个函数的进程不是进程组长, 会话创建成功
- 这个进程会变成当前会话中的第一个进程,同时也会变成新的进程组的组长
- 该函数调用成功之后, 当前进程就脱离了控制终端,因此不会阻塞终端
三、创建守护进程
如果要创建一个守护进程,标准步骤如下,部分操作可以根据实际需求进行取舍:
-
创建子进程, 让父进程退出
- 因为父进程有可能是组长进程,不符合条件,也没有什么利用价值,退出即可
- 子进程没有任何职务, 目的是让子进程最终变成一个会话, 最终就会得到守护进程
-
通过子进程创建新的会话,调用函数
setsid
,脱离控制终端, 变成守护进程 -
改变当前进程的工作目录(非必须)
-
修改当前进程的工作目录需要调用函数 chdir
-
int chdir(const char *path);
-
-
重新设置文件的掩码(非必须)
-
掩码: umask, 在创建新文件的时候需要和这个掩码进行运算, 去掉文件的某些权限
-
mode_t umask(mode_t mask);
-
-
关闭/重定向文件描述符(非必须)
-
标准输入,标准输出,标准错误,这三个文件描述符对应的都是当前终端,由于执行
setsid
调用,当前进程已经脱离了终端,因此关联的文件描述符也就没用了,可以关闭 -
close(STDIN_FILENO);close(STDOUT_FILENO);close(STDERR_FILENO);
-
重定向文件描述符(和关闭二选一): 改变文件描述符关联的默认文件, 让他们指向一个特殊的文件/dev/null,只要把数据扔到这个特殊的设备文件中, 数据被被销毁了
-
int fd = open("/dev/null", O_RDWR);dup2(fd, STDIN_FILENO);dup2(fd, STDOUT_FILENO);dup2(fd, STDERR_FILENO);
-
四、守护进程应用
每隔两秒,将当前系统时间写入到文件中
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <signal.h>
#include <sys/time.h>
#include <time.h>// 信号的处理动作
void writeFile(int num) {// 得到系统时间time_t seconds = time(NULL);// 时间转换, 总秒数 -> 可以识别的时间字符串struct tm* loc = localtime(&seconds);// sprintf();char* curtime = asctime(loc);// 文件权限 0664 & ~022int fd = open("./time.log", O_WRONLY|O_CREAT|O_APPEND, 0664);// 写入文件write(fd, curtime, strlen(curtime));// 关闭文件close(fd);
}int main() {pid_t pid = fork();if(pid > 0) {// 父进程exit(0);}// 子进程成为守护进程setsid();// 修改进程的工作目录chdir("/home/lyj");// 设置掩码, 在进程中创建文件的时候这个掩码就起作用了umask(022);// 重定向和终端关联的文件描述符int fd = open("/dev/null", O_RDWR);dup2(fd, STDIN_FILENO);dup2(fd, STDOUT_FILENO);dup2(fd, STDERR_FILENO);// 委托内核捕捉并处理将来发生的信号-SIGALRM(14)struct sigaction act;act.sa_flags = 0;act.sa_handler = writeFile;sigemptyset(&act.sa_mask);sigaction(SIGALRM, &act, NULL);// 设置定时器struct itimerval val;val.it_value.tv_sec = 2;val.it_value.tv_usec = 0;val.it_interval.tv_sec = 2;val.it_interval.tv_usec = 0;setitimer(ITIMER_REAL, &val, NULL);while(1) {sleep(10000);}return 0;
}
可以看到,这个程序已经在运行了