守护进程的概念:
- 运行在后台的一种特殊进程.
- 不随着 用户的注销 而退出.
- 独立于终端,不能直接和用户进行交互.
- 周期性的执行某种任务,或者等待某些发送的事件.
创建守护进程的方法:
void mydaemon(void)
{
pid_t pid;
int fd, i, nfiles;
struct rlimit rl;
pid = fork();
if(pid < 0)
ERROR_EXIT("First fork failed!");
if(pid > 0)
exit(EXIT_SUCCESS);// father exit
if(setsid() == -1)
ERROR_EXIT("setsid failed!");
pid = fork();
if(pid < 0)
ERROR_EXIT("Second fork failed!");
if(pid > 0)// father exit
exit(EXIT_SUCCESS);
#ifdef RLIMIT_NOFILE
/* 关闭从父进程继承来的文件描述符 */
if (getrlimit(RLIMIT_NOFILE, &rl) == -1)
ERROR_EXIT("getrlimit failed!");
nfiles = rl.rlim_cur = rl.rlim_max;
setrlimit(RLIMIT_NOFILE, &rl);
for(i=3; i<nfiles; i++)
close(i);
#endif
/* 重定向标准的3个文件描述符 */
if(fd = open("/dev/null", O_RDWR) < 0)
ERROR_EXIT("open /dev/null failed!");
for(i=0; i<3; i++)
dup2(fd, i);
if(fd > 2) close(fd);
/* 改变工作目录和文件掩码常量 */
chdir("/");
umask(0);
}
解读:
-
成为后台进程,用fork创建子进程,父进程退出,子进程成为孤儿进程被init接管,子进程变为后台进程。
-
脱离父进程的控制终端,登陆会话和进程组.调用setsid()让子进程成为新会话的组长,脱离父进程的会话期。setsid()在调用者是某进程组组长时会失败,但是A已经保证了子进程不会是组长,B之后子进程变成了新会话组的组长。
-
禁止进程重新开启控制终端.因为会话组的组长有权限重新打开控制终端,所以这里第二次fork将子进程结束,留着孙进程,孙进程不是会话组的组长所以没有权利再打开控制终端,这样整个程序就与控制终端隔离了。
-
关闭文件描述符.进程从创建它的父进程那里继承了打开的文件描述符。如不关闭,将会浪费系统资源,造成进程所在的文件系统无法卸下以及引起无法预料的错误。
-
重定向0,1,2标准文件描述符.将三个标准文件描述符定向到/dev/null中
-
改变工作目录和文件掩码.进程活动时,其工作目录所在的文件系统不能卸下(比如工作目录在一个NFS中,运行一个daemon会导致umount无法成功)。一般需要将工作目录改变到根目录。对于需要转储核心,写运行日志的进程将工作目录改变到特定目录如chdir("/tmp"),进程从创建它的父进程那里继承了文件创建掩模。它可能修改守护进程所创建的文件的存取位。为防止这一点,将文件创建掩模清除:umask(0);