概述
守护进程是在需要在后台长期运行不受终端控制的进程,通常情况下守护进程在系统启动时自动运行,在服务器关闭的时候自动关闭;守护进程的名称通常以d结尾,比如sshd、xinetd、crond、atd等。
守护进程编程规则
调用umask将文件模式创建屏蔽字设置为一个已知值(通常是0)
调用fork(),创建新进程,它会是将来的守护进程
然后使父进程exit,保证子进程不是进程组组长
调用setsid创建新的会话
会话:是一个或者多个进程组的集合,通常一个会话开始与用户登录,终止于用户退出。在此期间,该用户运行的所有进程都属于这个会话期。
将进程的当前目录改为根目录 (如果把当前目录作为守护进程的目录,当前目录不能被卸载,它作为守护进程的工作目录了。)
关闭不再需要的文件描述符
将标准输入、标准输出、标准错误重定向到/dev/null
setsid
pid_t setsid(void);
setsid() creates a new session if the calling process is not a process group leader. The calling process is the leader of the new session, the process group leader of the new process group, and has no controlling terminal. The process group ID and session ID of the calling process are set to the PID of the calling process. The calling process will be the only process in this new process group and in this new session.
/*当调用进程不是一个进程组的组长时,Setsid创建一个新的会话;调用者进程会是这个会话期唯一的一个进程,且是该进程组的组长;调用者进程id是组id,也是会话期的id。不能用进程组组长去调用setsid函数*/
//示例: int main() { pid_t pid = fork(); if (pid == -1) err_exit("fork error"); else if (pid != 0) exit(EXIT_SUCCESS); // //查看下面这一部分代码在注释的前后有什么变化 // pid_t id = setsid(); // if (id == -1) // err_exit("setsid error"); // else // cout << "new session id = " << id << endl; cout << "getpid = " << getpid() << endl; cout << "getpgid = " << getpgid(getpid()) << endl; return 0; }
RETURN VALUE
On success, the (new) session ID of the calling process is returned. On error, (pid_t) -1 is returned, and errno is set to indicate the error.
Linux中的守护进程API
int daemon(int nochdir, int noclose);
参数:
nochdir:=0将当前目录更改至“/”
noclose:=0将标准输入、标准输出、标准错误重定向至“/dev/null”
DESCRIPTION
The daemon() function is for programs wishing to detach themselves from the controlling terminal and run in the background as system daemons. If nochdir is zero, daemon() changes the calling process‘s current working directory to the root directory ("/"); otherwise, the current working directory is left unchanged. If noclose is zero, daemon() redirects standard input, standard output and standard error to /dev/null; otherwise, no changes are made to these file descriptors.
//示例:自己动手写daemon函数(一个比较简单的示例) bool myDaemon(bool nochdir, bool noclose) { umask(0); pid_t pid = fork(); if (pid == -1) err_exit("fork error"); else if (pid != 0) //parent exit(0); setsid(); if (nochdir == 0) chdir("/"); if (noclose == 0) { int i; for (i=0; i < 3; ++i) close(i); open("/dev/null", O_RDWR); //相当于把0号文件描述符指向/dev/null dup(0); //把0号文件描述符 赋值给空闲的文件描述符 1 dup(0); //把0号文件描述符 赋值给空闲的文件描述符 2 } return true; } //测试 int main(int argc, char *argv[]) { myDaemon(0, 0); //0表示做出改变(当前目录,文件描述符),1表示不改变 printf("test ...\n"); while (true) { sleep(1); } return 0; }
//一个比较经典和完善的实例;来自《APUE》 #include "apue.h" #include <syslog.h> #include <fcntl.h> #include <sys/resource.h> bool myDaemon(const char *cmd); int main(int argc, char *argv[]) { myDaemon("xiaofang"); while (true) { sleep(1); } return 0; } bool myDaemon(const char *cmd) { umask(0); //Get maximum number of file descriptors. rlimit rlt; if (getrlimit(RLIMIT_NOFILE,&rlt) < 0) { err_quit("%s: can‘t get file limit",cmd); } //Become a session leader to lose controlling TTY. pid_t pid = fork(); if (pid == -1) { err_quit("%s: can‘t fork",cmd); } if (pid != 0) //parent { exit(0); } setsid(); //Ensure future opens won‘t allocate controlling TTYs. struct sigaction sa; sa.sa_handler = SIG_IGN; sigemptyset(&sa.sa_mask); sa.sa_flags = 0; if (sigaction(SIGHUP,&sa,NULL) < 0) { err_quit("%s can‘t ignore SIGHUP",cmd); } if ((pid = fork()) < 0) { err_quit("%s: can‘t fork",cmd); } else if (pid != 0) //Second Parent { exit(EXIT_SUCCESS); } //change the current working directory to the root if (chdir("/") < 0) { err_quit("%s: can‘t change directory to /",cmd); } //close all open file description if (rlt.rlim_max == RLIM_INFINITY) { rlt.rlim_max = 1024; } for (unsigned int i = 0; i < rlt.rlim_max; ++i) { close(i); } //Attach file descriptors 0, 1, and 2 to /dev/null. int fd0 = open("/dev/null",O_RDWR); int fd1 = dup(0); int fd2 = dup(0); //Initialize the log file. openlog(cmd,LOG_CONS,LOG_DAEMON); if (fd0 != 0 || fd1 != 0 || fd2 != 0) { syslog(LOG_ERR,"unexpected file descriptors %d %d %d", fd0,fd1,fd2); exit(EXIT_FAILURE); } return true; }