什么是精灵进程
精灵进程也称守护进程(Daemon):是运行在后台的一种特殊进程,它独立于控制终端并周期性的执行某种任务,或等待处理某些发生的事件。Linux大多数服务器就是用精灵进程实现的。
1.我们先来观察一些系统中的精灵进程
精灵进程不具有控制终端所以TTY-?,终端前台进程组的id为-1。自己写一个精灵进程
根据精灵进程的概念,我们可以总结出实现一个精灵进程的基本步骤。
1.调用umask将文件模式创建屏蔽字设为0
2.调用fork()父进程退出
3.调用setsid创建一个新会话
4.再次fork()父进程退出
5.将当前工作目录更改为根目录
6.关闭不需要的文件描述符
7.忽略SIGCHLD信号
接下来让我们一步一步来实现一个精灵进程。
1.调用umask将文件模式创建屏蔽字设为0
umask(0);
由继承得来的文件方式创建屏蔽字可能会拒绝设置某些许可权。在这种情况下如果精灵进程想要创建一个可读可写的文件,而文件方式创建屏蔽字屏蔽了读写的许可权,那么精灵进程创建的文件就无法获得读写权限。
2.调用fork()父进程退出
pid_t id = fork();
if(id < 0)
exit(EXIT_FAILURE);
if(id > 0)
exit(EXIT_SUCCESS);
精灵进程没有控制终端,为了不使其成为孤儿进程需要调用setsid()函数获得一个SID。
执行setsid()函数的结果
1.创建一个新的会话,当前进程成为会话首进程,SID就是该进程的pid。
2.创建一个新的进程组,当前进程成为组长进程PGID就是该进程的pid。
3.如果当前进程原本有一个控制终端,则它失去这个控制终端。
4.调用该函数成功返回一个新的会话id失败返回-1,调用该函数之前该进程不能是该进程组的组长进程。
为了满足性质4我们可以调用fork创建一个子进程,子进程不是进程组的第一个进程所以一定不会是组长进程,我们用子进程调用setsid()函数。
3.调用setsid创建一个新会话
pid_t sid = setsid();
if(sid < 0)
exit(EXIT_FAILURE);
4.再次fork()父进程退出
id = fork();
if(id < 0)
exit(EXIT_FAILURE);
if(id != 0)
exit(EXIT_SUCCESS);
会话首进程仍然可以打开一个终端,为了保证后续不不会再打开终端,再次fork()保证精灵进程不再是会话首进程。
5.将当前工作目录更改为根目录
if(chdir("/") < 0)
exit(EXIT_FAILURE);
从父进程继承过来的工作目录,可能在一个装配的文件系统中,因为精灵进程通常在系统中是一直存在的,如果当前工作目录在一个装配的文件系统中,那么该文件系统不能被拆卸。
6.关闭不需要的文件描述符
close(0);
close(1);
close(2);
这样精灵进程不在持有从父进程继承来的文件描述符。
7.忽略SIGCHLD信号
signal(SIGCHLD,SIG_IGN);
这样fork出来的子进程在执行完毕后自动清理不会产生僵尸进程。
- 完整代码
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<signal.h>
void Daemon()
{
umask(0);
pid_t id = fork();
if(id < 0)
exit(EXIT_FAILURE);
if(id > 0)
exit(EXIT_SUCCESS);
pid_t sid = setsid();
if(sid < 0)
exit(EXIT_FAILURE);
signal(SIGCHLD,SIG_IGN);
id = fork();
if(id < 0)
exit(EXIT_FAILURE);
if(id != 0)
exit(EXIT_SUCCESS);
if(chdir("/") < 0)
exit(EXIT_FAILURE);
close(0);
close(1);
close(2);
}
int main()
{
Daemon();
while(1);
return 0;
}