在正常条件下,我们将程序运行产生的信息打印到控制台实时显示,如果我们想讲一个程序以守护进程的方式进行运行,就需要改变信息的输出方向,将其导向到配置文件里设置的日志文件。

  将一个进程转换为守护进程需要进行几个步骤:

  1.fork一个新的进程,将父进程退出。

  2.将0、1、2三个文件描述符重定向。

  3.将1重定向到日志文件。

void daemonize()
{
           if(fork() != 0)     //将父进程退出
                      exit(0);
           setsid();            //设置新的会话
           LOG(LOG_LEVEL_INFO, "Daemonized ... pid = %d", (int)getpid());
 
           int fd = 0;   
           if((fd = open("/dev/null", O_RDWR, 0)) != -1)  //打开空设备文件
           {
                      dup2(fd, STDIN_FILENO);         //将0重定向到/dev/null
                      dup2(fd, STDOUT_FILENO);      //将1重定向到/dev/null
                      dup2(fd, STDERR_FILENO);      //将2重定向到/dev/null
                      if(fd > STDERR_FILENO)
                                   close(fd);
           }

           if(g_conf->log_file != NULL && (fd = open(g_conf->log_file, O_RDWR|O_APPEND|O_CREAT, 0))!= -1)
           {
                      dup2(fd, STDOUT_FILENO);      // 将2重定向到日志文件
                      if(fd>STDERR_FILENO)
                                  close(fd);
           }
}

  其中dup2函数的作用是负复制文件描述符,将第一个参数中的文件描述符复制到第二个参数的文件描述符,即第二个文件描述符指向第一个文件描述符指向的文件,这意味着,这两个描述符共享同一个数据结构file结构体,这个file结构体的引用计数为2。