一·基本概念
(1) 守护进程是生存期长的一种进程。常常在系统引导装入时启动,仅在系统关闭时才终止。因为没有控制终端,所以说是在后台运行的
守护进程不受用户登录和销毁的控制,通常以d结尾表示为daemon,守护进程是孤儿进程。当前作业与终端无关(因为守护进程自成一个会话,所以不受终端控制)
二 相关指令 ps -axj
-a 显示当前用户其他用户所拥有的进程状态
-x 显示有和没有终端控制的进程状态(TPGID一栏写着 -l 的是没有终端控制的进程)
-j 显示与作业有关的信息:会话ID,进程组ID,控制终端以及终端进程组ID。
在COMMAND一列用[]括起来的表示内核线程。
三 创建守护进程
创建守护进程要调用setsid函数。(这里我们需要注意是fork()之后再创建守护进程,让子进程调用setsid,父进程退出)因为要保证调用者不是进程组的组长进程。 函数调用成功返回当前进程ID,失败返回-1.
创建成功结果:(1)创建了一个新的会话,成为会话首进程,当前进程ID为会话ID。
(2)创建了一个新的进程组,并成为进程组的组长进程,当前进程ID就是进程组ID。
(3) 与终端去关联
创建步骤:
(1)调用umask将文件模式屏蔽字设置为0
(2)调用fork,父进程exit。(保证了子进程不是进程组的组长进程)
(3)调用setsid创建一个新会话(成为新会话首进程,成为一个新进程组的组长进程,没有控制终端)
(4)将当前工作目录更改为更目录(若是不修改,则目录有可能被删除)
(5)关闭不需要的文件描述符表
(6)忽略SIGCHLD信号
fork两次的问题:
有人建议再次调用fork(及fork两次的问题)。这样可以保证守护进程不是会话首进程,按照System V规则中,这样可以防止它和取得终端控制。
创建守护进程代码
#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<signal.h>
void creat_daemon()
{
umask(0);
if(fork()>0)
{
exit(0);
}
setsid();
chdir("/");
close(0);
close(1);
close(2);
signal(SIGCHLD,SIG_IGN);
}
int main()
{
creat_daemon();
while(1)
{
}
return 0;
}