什么是守护进程daemon
Linux daemon是运行于后台常驻内存的一种特殊进程,周期性的执行或者等待trigger执行某个任务,与用户交互断开,独立于控制终端。一个守护进程的父进程是init进程,它是一个孤儿进程,没有控制终端,所以任何输出,无论是向标准输出设备stdout还是标准出错设备stderr的输出都被丢到了/dev/null中。守护进程一般用作服务器进程,如httpd,syslogd等。
实例(创建一个daemon,每隔10秒向/mydaemon.log文件写入当前时间一共三次)
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);
}
A(7~12行):成为后台进程:
用fork创建子进程,父进程退出,子进程成为孤儿进程被init接管,子进程变为后台进程。
B(14~15行):脱离父进程的控制终端,登陆会话和进程组
调用setsid()让子进程成为新会话的组长,脱离父进程的会话期。setsid()在调用者是某进程组组长时会失败,但是A已经保证了子进程不会是组长,B之后子进程变成了新会话组的组长。
C(17~22行):禁止进程重新开启控制终端
因为会话组的组长有权限重新打开控制终端,所以这里第二次fork将子进程结束,留着孙进程,孙进程不是会话组的组长所以没有权利再打开控制终端,这样整个程序就与控制终端隔离了。
D(23~31行):关闭文件描述符
进程从创建它的父进程那里继承了打开的文件描述符。如不关闭,将会浪费系统资源,造成进程所在的文件系统无法卸下以及引起无法预料的错误。
E(32~36行):重定向0,1,2标准文件描述符
将三个标准文件描述符定向到/dev/null中
F(38~40行):改变工作目录和文件掩码
进程活动时,其工作目录所在的文件系统不能卸下(比如工作目录在一个NFS中,运行一个daemon会导致umount无法成功)。一般需要将工作目录改变到根目录。对于需要转储核心,写运行日志的进程将工作目录改变到特定目录如chdir("/tmp"),进程从创建它的父进程那里继承了文件创建掩模。它可能修改守护进程所创建的文件的存取位。为防止这一点,将文件创建掩模清除:umask(0);
注:D,E,F三步是对当前工作环境的修改,可以先做,因为这些修改都会被子进程继承下来