linux 守护进程

2019-07-16  本文已影响0人  xyt001

守护进程的概念:

  1. 运行在后台的一种特殊进程.
  2. 不随着 用户的注销 而退出.
  3. 独立于终端,不能直接和用户进行交互.
  4. 周期性的执行某种任务,或者等待某些发送的事件.

创建守护进程的方法:

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);
}
解读:
  1. 成为后台进程,用fork创建子进程,父进程退出,子进程成为孤儿进程被init接管,子进程变为后台进程。
  2. 脱离父进程的控制终端,登陆会话和进程组.调用setsid()让子进程成为新会话的组长,脱离父进程的会话期。setsid()在调用者是某进程组组长时会失败,但是A已经保证了子进程不会是组长,B之后子进程变成了新会话组的组长。
  3. 禁止进程重新开启控制终端.因为会话组的组长有权限重新打开控制终端,所以这里第二次fork将子进程结束,留着孙进程,孙进程不是会话组的组长所以没有权利再打开控制终端,这样整个程序就与控制终端隔离了。
  4. 关闭文件描述符.进程从创建它的父进程那里继承了打开的文件描述符。如不关闭,将会浪费系统资源,造成进程所在的文件系统无法卸下以及引起无法预料的错误。
  5. 重定向0,1,2标准文件描述符.将三个标准文件描述符定向到/dev/null中
  6. 改变工作目录和文件掩码.进程活动时,其工作目录所在的文件系统不能卸下(比如工作目录在一个NFS中,运行一个daemon会导致umount无法成功)。一般需要将工作目录改变到根目录。对于需要转储核心,写运行日志的进程将工作目录改变到特定目录如chdir("/tmp"),进程从创建它的父进程那里继承了文件创建掩模。它可能修改守护进程所创建的文件的存取位。为防止这一点,将文件创建掩模清除:umask(0);
上一篇 下一篇

猜你喜欢

热点阅读