守护进程的概念:
守护进程(daemon)是指在UNIX或其他多任务操作系统中在后台执行的电脑程序,并不会接受电脑用户的直接操控。此类程序会被以进程的形式初始化。守护进程程序的名称通常以字母“d”结尾:例如,syslogd就是指管理系统日志的守护进程。
通常,守护进程没有任何存在的父进程(即PPID=1),且在UNIX系统进程层级中直接位于init之下。守护进程程序通常通过如下方法使自己成为守护进程:对一个子进程调用fork,然后使其父进程立即终止,使得这个子进程能在init下运行。这种方法通常被称为“脱壳”。
守护进程的创建:
1)首先要做的是调用umask将文件模式创建屏蔽字设置为0。由继承得来的文件模式创建屏蔽字可能会拒绝设置某些权限。
2)调用fork,然后使父进程退出(exit)。这样做实现了下面几点:第一,如果该守护进程时作为一条简单shell命令启动的,那么父进程终止使得shell认为这条命令已经执行完毕;第二,子进程继承了父进程的进程组ID,但具有一个新的进程ID,这就保证了子进程不是一个进程组的组长进程。(这对于下面要做的setsid调用是必要的前提条件)
3)调用setsid创建一个新会话,使调用进程:成为新会话的首进程;成为一个新进程组的组长进程;没有控制终端
4)将当前工作目录更改为根目录。从父进程处继承过来的当前工作目录可能在一个装配文件系统中,如果守护进程工作在这样的文件系统中,那么该文件系统就不能被卸载。
5)关闭不再需要的文件描述符。这使守护进程不再持有从其父进程继承来的某些文件描述符。
6)某些守护进程打开/dev/null使其具有文件描述符0、1和2,这样,任何一个试图读标准输入、写标准输出或标准出错的库例程都不会产生任何效果。
代码的实现:
#include<stdio.h>
#include<unistd.h>
#include<signal.h>
#include<stdlib.h>
#include<fcntl.h>
#include<sys/stat.h>
void daem()
{
umask(0);//设置umask为0;
pid_t pid=fork();//创建一个子进程
if(pid<0)
{
perror("错误!");
return;
}
else if(pid>0)
{
exit(0);
}
setsid();
if(chdir("/")<0)
{
perror("chdir error~");
return;
}
//关闭文件描述符
close(0);//关闭stdin
close(1);//关闭stdout
close(2);//关闭stderror
signal(SIGCHLD,SIG_IGN);
}
int main()
{
daem();
while(1);
return 0;
}
fork两次的原因:
(1)调用一次fork的作用:
为了让shell认为该命令已经终止,为后面的setsid服务,调用setsid函数的进程不能是进程组长,所以需要fork出子进程。
(2)第二次调用的作用:
虽然当前关闭了和终端的联系,但是后期会误操作再一次打开了终端,于是再fork一次,将父进程退出,再次fork子进程守护进程继续运行,保证了该精灵进程不是对话期的首进程。