守护进程

什么是守护进程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三步是对当前工作环境的修改,可以先做,因为这些修改都会被子进程继承下来

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。