设置守护进程

一. 什么是守护进程
守护进程(daemon)是一类在后台运行的特殊进程,用于执行特定的系统任务。很多守护进程在系统引导的时候启动,并且一直运行直到系统关闭。另一些只在需要的时候才启动,完成任务后就自动结束。

在linux系统中,我们会发现在系统启动的时候有很多的进程就已经开始跑了,也称为服务,这也是我们所说的守护进程。

守护进程是脱离于终端并且在后台运行的进程,脱离终端是为了避免在执行的过程中的信息在终端上显示,并且进程也不会被任何终端所产生的终端信息所打断。

守护进程一般的生命周期是系统启动到系统停止运行,也可以通过杀死进程的方式来结束进程的生命周期。linux系统中有很多的守护进程,最典型的就是我们经常看到的服务进程。当然,我们也经常会利用守护进程来完成很多的系统或者自动化任务。

二. 为什么要设置守护进程

三. 设置守护进程的步骤
创建守护进程的步骤:

(1)创建子进程,父进程退出

因为守护进程是脱离终端控制的,所以要造成一种在终端里已经运行完的假象,把所有的工作都放在子进程中去完成。我们知道,父进程退出后,子进程其实就是变成了孤儿进程,孤儿进程一般是由1号进程收养,也就是我们所谓的init进程,也就是说原来的子进程变成了init的子进程。

(2)创建会话

首先看一下进程组的概念:

进程组 :

1.每个进程也属于一个进程组

2.每个进程主都有一个进程组号,该号等于该进程组组长的PID号 .

3.一个进程只能为它自己或子进程设置进程组ID号

会话期:会话期(session)是一个或多个进程组的集合。

setsid()函数可以建立一个对话期:

如果,调用setsid的进程不是一个进程组的组长,此函数创建一个新的会话期。

(1)此进程变成该对话期的首进程

(2)此进程变成一个新进程组的组长进程。

(3)此进程没有控制终端,如果在调用setsid前,该进程有控制终端,那么与该终端的联系被解除。 如果该进程是一个进程组的组长,此函数返回错误。

(4)为了保证这一点,我们先调用fork()然后exit(),此时只有子进程在运行
其实就是起到了进程脱离原会话的控制,摆脱了原进程组的控制,摆脱了原控制终端的控制。
那么为什么要调用setsid函数呢?主要是因为前面创建子进程的时候,随人父进程退出了,但是子进 程拷贝了父进程的会话、进程组、控制终端等等,所以需要使用setsid来摆脱控制。

(3)改变当前目录为根目录
在fork子进程的时候,子进程也继承了父进程的工作目录。通常是让‘/’目录作为守护进程的当前目录,可以避免很多麻烦。

(4)设置文件掩码
同一个文件的权限掩码是一样的,我们使用fork创建的子进程会从父进程那里继承一些已经打开的文件,打开的文件可能永远不会被守护进程读写,但是他们一样消耗资源,而且可能导致所在的文件系统无法结束。

用例1

#include <iostream>
#include <unistd.h>
#include <fstream>
#include <fcntl.h>

void daemonize()
{
    if(fork() != 0)     //将父进程退出
        exit(0);

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

    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 ((fd = open("test.log", O_RDWR|O_APPEND|O_CREAT, 0))!= -1) {
        dup2(fd, STDOUT_FILENO);      // 将2重定向到日志文件
        if(fd>STDERR_FILENO)
            close(fd);
    }
}

int main()
{
    daemonize();
    while(1) {
    }
    return 0;
}

用例2

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

#define MAXFILE 65535

int main()
{
    int i,fd;
    pid_t pid;
    char *buf = "hello,dameon";
    int len = strlen(buf);

//第一步,创建子进程
    pid = fork();

    if (pid < 0) {
        printf("error\n");
        exit(1);//异常退出
    }
    else if (pid > 0) {
       exit(0);
    }
//第二步,创建会话
    setsid();

//第三步,修改根目录
    chdir("/");

//第四步,设置掩码
    umask(0);
    for (i=0;i<MAXFILE;i++) {
        close(i);
    }

    while(1) {
        fd = open("/mnt/hgfs/H/C/dameon.log",O_CREAT|O_WRONLY,0600);
        if(fd < 0) {
            perror("open");
            exit(1);
        }
        write(fd,buf,len+1);
        close(fd);
    }
    return 0;
}

参考:
[1] https://zhuanlan.zhihu.com/p/56840430

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值