• Linux进程实践(5) --守护进程


    概述

       守护进程是在需要在后台长期运行不受终端控制的进程,通常情况下守护进程在系统启动时自动运行,在服务器关闭的时候自动关闭;守护进程的名称通常以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 ...
    ");
        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;
    }

  • 相关阅读:
    oracle12.2RAC之OGG安装配置(一)
    【java框架】JPA(3) -- JPA映射关系
    【java框架】JPA(2) -- JPA基础
    【java框架】Struts2(3) -- Struts2结果视图及参数接收
    【问题管理】-- Struts2配置struts.xml中Action访问报There is no Action mapped for namespace...
    【java框架】Struts2(2) -- Struts2配置
    JavaScript(1) -- JS入门
    【java框架】MyBatis(2)--Mapper代理接口及批量操作
    【java框架】MyBatis(1)--MyBatis入门
    C# AutoMapper 了解一下
  • 原文地址:https://www.cnblogs.com/itrena/p/5926972.html
Copyright © 2020-2023  润新知