UNIX下的网络服务程序,如Web Server,FTP,Telnet一般都是由守护进程(Daemon)来实现的。守护进程不占用终端,在后台运行。UNIX的守护进程一般都命名为 *d 的形式,如httpd,telnetd等等。其实,守护进程的实现是非常简单的,在我的程序中,我使用一个INIT_DAEMON宏来实现守护进程的初始化工作,如图2.4所示。第一次调用fork函数,为避免挂起控制终端将守护进程放入后台执行。然后调用setsid函数脱离控制终端,登录会话和进程组,使该进程成为会话组长,与原来的登录会话和进程组脱离,进程同时与控制终端脱离。进程已经成为无终端的会话组长。但它可以重新申请打开一个控制终端。可以通过使进程不再成为会话组长来禁止进程重新打开控制终端,这就需要第二次调用fork函数,父进程(会话组长)退出,子进程继续执行,并不在拥有打开控制终端的能力。在正在执行的进程中调用INIT_DAEMON后 进程将成
#define INIT_DAEMON \
{ \
if( fork() > 0 ) exit(0); \
setsid(); \
if( fork() > 0 ) exit(0); \
}
为守护进程,脱离控制终端进入后台执行。比如,我们的网络服务程序,可以在完成创建套接口,绑定套接口,设置套接口为监听模式后,变成守护进程进入后台执行而不占用控制终端,这是网络服务程序的常用模式。
INIT_DAEMON宏只提供了很简单的功能,守护进程一旦脱离了终端,退出就成了问题。我目前使用 PS 查出进程ID然后 kill 之,笨得实在可以。以后进一步想出一种更好的守护进程退出机制,有办法的朋友,请不吝赐教写信给我,我会帖出来的。
方法如下:
Linux下守护进程的创建有很多的方法,比如我们可以使用cron,inetd等程序来创建.这里介绍在控制终端上有用户来启动的守护程序.这种守护程序不依赖于任何一个终端,不会随着用户的退出而结束.这种程序经常用于网络程序之中.
将一个程序变为守护程序一般按照下面的步骤.
调用函数fork,然后父进程推出,这样子进程就变为了后台进程了.同时子进程不成为进程组的组长(组长可能是父进程或者是创建父进程的进程)为第二步系统调用setsid做准备.
调用setsid创建一个新的会议组,进程成为一个新的会议组的组长.这样这个会议组就没控制终端了.
添加信号SIGHUP的处理.为后面的会议组长退出作出处理.同时再一次调用fork.然后父进程推出.由于进程组长退出时向所有会议成员发出SIGHUP,所以我忽略这个信号.这样我们的这个进程就没有了控制终端了.
调用函数chdir将进程的工作目录改到根目录(/).这样我们的程序可以不占用某个可卸载的设备.防止root卸载设备时系统报告设备忙.
关闭所有的文件描述符,同时重定向3个标准文件描述符.
下面的程序将创建一个守护程序,这个程序重定向了标准输出,标准输入,和标准错误输出. 每分钟向标准错误输出可用的系统内存页数.
int main(int argc,char **argv)
{
struct sigaction act;
int error,in,out;
time_t now;
int memory;
/* 父进程退出 */
if(fork()!=0) exit(1);
/* 创建一个新的会议组 */
if(setsid()<0)exit(1);
/* 忽略信号SIGHUP */
act.sa_handler=SIG_IGN;
sigemptyset(&act.sa_mask);
act.sa_flags=0;
if(sigaction(SIGHUP,&act,NULL)==-1)exit(1);
/* 子进程退出,孙进程没有控制终端了 */
if(fork()!=0) exit(1);
if(chdir("/")==-1)exit(1);
/* 标准错误重定向 */
error=open("/tmp/error",O_WRONLY|O_CREAT,0600);
dup2(error,STDERR_FILENO);
close(error);
/* 标准输入重定向 */
in=open("/tmp/in",O_RDONLY|O_CREAT,0600);
if(dup2(in,STDIN_FILENO)==-1)perror("in");
close(in);
/* 标准输出重定向 */
out=open("/tmp/out",O_WRONLY|O_CREAT,0600);
if(dup2(out,STDOUT_FILENO)==-1)perror("out");
close(out);
while(1)
{
time(&now);
memory=sysconf(_SC_AVPHYS_PAGES);
fprintf(stderr,"时间\t%s\t\t可用内存[%d]\n",ctime(&now),memory);
sleep(60);
}
exit(0);
}
运行这个程序后,使用ps -x命令可以看到这个程序的tty是?.表示这个程序没有控制终端.