概述
操作系统总有一些默默在后台运行着的程序,被称作服务,不与使用者进行交互。他们的功能很简单一般是重复性周期性的或者只有等待某些事件的发生才会做出相应动作,例如邮件接收监听程序每隔一段时间就会扫描一下是否有新的邮件到来,
linux
中同样存在这样的程序,在后台运行脱离终端控制,不在终端进行交互,这样的进程被称作守护进程,这样的进程不受用户注销影响
通俗解释
(这是一个粗糙的解释):一个进程想成为守护进程,需要从所在进程组(包含多个进程的集合)脱离出来,同时该进程不能是进程组组长。什么是进程组组长?进程组组长就是该进程组的第一个进程,一个进程创建了一个子进程,那么该进程组就有两个成员,一个是父进程,一个是子进程,那么父进程就是该进程组的组长。如果子进程想成为守护进程,就需要从进程组脱离出来,将身份提高变身为会长,什么是会(会就是会话即多个进程组的集合),子进程从进程组脱离出来变身会长(该会长虽然只有自己一个成员),身份不同地位待遇就有所不同,便有了变身守护进程的可能。而原来的进程组组长不能变更身份,否则造成地位错乱。
具体方法
- 创建一个会话的注意事项
- 不能是进程组组长
- 创建会话的进程称为新的进程组的组长
- 有些
linux
版本需要root
权限执行此操作(ubuntu
不需要) - 创建出的新会话会丢弃原有的控制终端
- 一般步骤:父进程死,子进程执行创建会话操作(
setsid
)
- 获取进程所属会话ID
pid_t getsid(void)
- 创建一个会话
pid_t setsid(void)
创建守护进程模型
fork
子进程,父进程退出
- 必须进行的操作
- 子进程创建新会话
- 必须进行的操作
- 改变当前工作目录
- 只是为了保证程序健壮性,防止意外发生,例如在
U
盘目录中启动a.out
,在启动过程中,U
盘拔掉了,造成运行错误。 - 非必须操作
- 只是为了保证程序健壮性,防止意外发生,例如在
- 重设文件掩码
- 子进程会继承父进程的掩码
- 可增加子进程程序操作的灵活性
umask(0)
- 根据实际情况决定非必须操作
- 关闭文件描述符
- 新会话脱离终端,不需要终端相关文件描述符
close(0)
close(1)
close(2)
- 释放资源
- 执行核心工作
- 运行守护进程所执行的操作
小示例
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <sys/stat.h>
#include <fcntl.h>
int main()
{
int pid;
pid=fork();
if(pid==0)
{
close(0);
close(1);
close(2);
umask(0);
setsid();
while(1)
{
}
}else
{
exit(0);
}
return 0;
}
程序运行结果:
root@DESKTOP-FR31BP0:/mnt/e/mycode/linux# gcc sh.c
root@DESKTOP-FR31BP0:/mnt/e/mycode/linux# ./a.out
root@DESKTOP-FR31BP0:/mnt/e/mycode/linux# ps aux
USER PID %CPU %MEM VSZ RSS TTY STAT START TIME COMMAND
root 1 0.0 0.0 10428 576 ? Ss 15:05 0:00 /init
root 2 0.0 0.0 26344 2292 tty1 Ss 15:05 0:00 /bin/bash
root 157 118 0.0 36596 172 ? Rs 15:19 0:03 ./a.out
root 158 0.0 0.0 43756 1824 tty1 R 15:19 0:00 ps aux
虽然子进程一直在运行,但并没有造成终端卡顿,使用ps aux
查看,可以看到a.out
确实在后台运行了。证明我们守护进程创建成功了