一、守护进程

守护进程,也叫精灵进程(daemon)

 

它和普通后台进程的区别在于以下三点

1、守护进程自成会话,而普通后台进程则不一定

2、守护进程不受终端的控制

3、守护进程就是后台进程,而后台进程不同于守护进程

 

 

 

用ps axj命令查看系统中的进程,TPGID一栏为 -1 的进程(这些进程没有控制终端)就是守护进程。

 

 

 

二、实现

创建守护进程的步骤如下:

1、调用umask把【文件模式创建屏蔽字】 设置为 0

    由于 umask 接收的参数会被取反,所以这个 0 传进去取反以后是最大的,也就是给该程序最大的权限

2、先fork,然后让父进程退出

    这是因为,要设置成守护进程的进程不能使原进程组中的组长进程,所以fork的目的是创建一个子进程,这样,这个子进程一定不会是组长进程,然后父进程就没什么用了。

3、调用 setid 创建一个新会话

    这时,这个进程会成为新会话的首进程(也是组长进程)

4、将当前工作目录改成根目录

5、关闭不需要的文件描述符

6、忽略SIGCHILD信号

 

 

实现代码如下:

wKiom1ePf2qwN4zXAAESa_0NhGo281.png

 

 

 

运行后,执行 ps axj

找到刚刚的进程

wKiom1ePgCPj6SgJAAAcjNOl6Jk435.png

 

同时还可以看到,它的父进程是 1 号进程,因此守护进程也是孤儿进程

 

不过这里要注意的是,实际使用的时候并不需要我们造这个轮子了 有现成的函数

daemon(0, 0) 就可以让当前的进程变为孤儿进程

 

 

三、fork()两次的原因

通常daemon的实现跟上面的实现方式还有一处不同:fork() 调用了两次,让后将第二个fork 创建出来的子进程作为守护进程。

 

这是因为,如果只fork一次的话那么有可能有这样的情况:

    第一次fork之后,父进程脱离,这时候子进程成为话首进程,子进程有可能会再打开一个控制终端,一旦这种事情发生,该进程就不再是守护进程了,因为它又一次受到终端的控制了

 

 

 

因此,如果再fork一次,那么这次得到的子进程将不可能是话首进程,因此不可能再打开终端了,所以通常用这个第二次fork出的子进程作为守护进程