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出来的子进程在执行完毕后自动清理不会产生僵尸进程。
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;
}
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)