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是?.表示这个程序没有控制终端.
|