守护进程(Daemon)是在后台运行的一种特殊进程,它脱离于终端,从而这可避免进程被任何终端所产生的信号打断,它在执行进程中的产生信息也不在任何终端上显示。守护进程周期性地执行某种任务或等待处理某些发生的事件,Linux的大多数服务器就是用守护进程实现的。
/* 处理可能的终端信号 */ signal(SIGTTOU, SIG_IGN); signal(SIGTTIN, SIG_IGN); signal(SIGTSTP, SIG_IGN); signal(SIGHUP , SIG_IGN);
/* 是父进程,结束父进程,子进程继续 */ if(fork()) exit(0);
基于以上原因,需要让为个子进程彻底摆脱该终端的影响,需要调用setsid()使子进程成为新的会话组长,代码如下:
setsid();
setsid()调用成功后,调用此函数的进程成为新的会话组长和新的进程组长,并与原来的进程组脱离关系。由于会话过程对控制终端的独占性,进程同时与控制终端脱离。
/* 结束第一子进程,第二子进程继续 */ if(fork()) exit(0);
#define NOFILE 256 for(i=0; i
chdir("/tmp");
umask(0);
/* 将子进程退出信号设为SIG_IGN,让系统帮助回收进程资源 */ signal(SIGCHLD, SIG_IGN);
整体代码如下:
#define NOFILE 256 void DaemonMode() { int num = 0; int fd0, fd1, fd2; /* 屏蔽可能的信号 */ signal(SIGTTOU, SIG_IGN); signal(SIGTTIN, SIG_IGN); signal(SIGTSTP, SIG_IGN); signal(SIGHUP , SIG_IGN); if(fork()) exit(0); setsid(); if(fork()) exit(0); chdir("/tmp/httpd"); umask(0); for(; num
如果调用进程已经是一个进程组的组长,则此函数返回错误。为了杜绝这种情况,通常先调用fork()创建子进程,然后使其父进程终止,而子进程继续,在子进程中调用此函数。
如果调用此函数的进程不是一个进程组组长,则此函数会创建一个新会话,调用setsid()函数的进程成为新的会话的领头进程,并与其父进程的会话组和进程组脱离。由于会话对控制终端的独占性,进程同时与控制终端脱离。
以上是剖析Linux的守护神的详细内容。更多信息请关注PHP中文网其他相关文章!