守护进程

2018-05-17  本文已影响0人  _codelover

守护进程


什么是守护进程

守护进程是生存期长的一种进程.它们常常在系统引导装入时启动,仅在系统关闭时才终止.因为它们没有控制终端,所以说它们是在后台运行的.unix系统有很多守护进程,它们执行日常事务活动. --- << APUE >>


linux下的守护进程

root@codelover:/home/codelover# ps -axj
PPID   PID  PGID   SID TTY      TPGID STAT   UID   TIME COMMAND
0     1     1     1 ?           -1 Ss       0   0:01 /sbin/init splash
0     2     0     0 ?           -1 S        0   0:00 [kthreadd](用于创建其他内核进程)
2     4     0     0 ?           -1 S<       0   0:00 [kworker/0:0H]
2     6     0     0 ?           -1 S<       0   0:00 [mm_percpu_wq]
2     7     0     0 ?           -1 S        0   0:00 [ksoftirqd/0]
2     8     0     0 ?           -1 S        0   0:01 [rcu_sched]
2     9     0     0 ?           -1 S        0   0:00 [rcu_bh]
2    10     0     0 ?           -1 S        0   0:00 [migration/0]
2    11     0     0 ?           -1 S        0   0:00 [watchdog/0]
2    12     0     0 ?           -1 S        0   0:00 [cpuhp/0]
2    13     0     0 ?           -1 S        0   0:00 [cpuhp/1]

如何编写守护进程

  1. 调用umask将文件模式创建屏蔽字设置为一个已知值(通常0),主要是设置程序创建文件时的默认权限

  2. 调用fork,然后退出父进程,这么做的原因是:

    • 守护进程需要脱离终端控制,那么需要成为一个新会话,成为新会话时才会切断和终端的联系
    • 而创建一个新会话的 前提条件是进程不是一个进程组的组长,因此让fork的子进程来做这个事,因为fork出来的子进程一定不是组长进程

Talk is cheap. Show me the code

#include <stdio.h>
#include <fcntl.h>
#include <unistd.h>
#include <sys/stat.h>
#include <sys/resource.h>
#include <syslog.h>


int main(int argc, char const *argv[]) {

    //将文件模式创建屏蔽字设置为0
    umask(0);
    pid_t pid = fork();

    if(pid < 0) {
        //创建失败
        return (0);
    } else if(pid != 0) {
        //结束父进程
        return (-1);
    }

    //设置新的会话id
    setsid();

    //重新fork,让守护进程不是不是会话首进程
    pid = fork();
    if(pid != 0) {
        return (-1);
    }

    //切换到根目录下
    chdir("/");

    //关闭所有打开的文件描述符
    struct rlimit r;
    getrlimit(RLIMIT_NOFILE, &r);
    for (size_t i = 0; i < r.rlim_max; i++) {
        close(i);
    }

    //重定向0,1,2
     int fd = open("/dev/null", O_RDWR);
     dup2(fd, 1);
     dup2(fd, 2);

     openlog("test.log", LOG_CONS, LOG_DAEMON);
     syslog(LOG_ERR, "%s","test error");

     while (1) {
         sleep(10);
     }
    return 0;
}


上一篇下一篇

猜你喜欢

热点阅读