Skip to the content.

homepage

DAEMON

终端

进程组

会话

实例

image

进程组、会话操作函数

守护进程

守护进程的创建步骤

  1. 执行一个 fork(),之后父进程退出,子进程继续执行
    • 不退出父进程,父进程死的时候会输出终端的提示符
    • fork可以确保daemon不会是当前任何一个进程组的首进程
    • 以成功让子进程建立新会话⬇️
  2. 子进程调用 setsid() 开启一个新会话
    • 这样此进程就是进程组和会话组的首进程
    • 这样就没有控制终端,成功脱离控制终端,但仍然有终端,确保内核不会为进程生成任何信号
    • 为什么不在父进程里开启新会话:
      • 若该父进程就是它所属会话的首进程,那所属会话组的id就是它的id
      • 新创建的会话组将会拥有相同的id
  3. 清除进程的 umask 以确保当守护进程创建文件和目录时拥有所需的权限。
  4. 修改进程的当前工作目录,通常会改为根目录(/)。
    • 若守护进程运行在U盘目录里,U盘就不能卸载了
    • 根目录一般不会卸载
  5. 关闭守护进程从其父进程继承而来的所有打开着的文件描述符。
    • 脱离了控制终端,但还是有终端
    • 不然会不小心操作,使得终端不小心显示数据
  6. 在关闭了文件描述符0、1、2之后,守护进程通常会打开/dev/null(丢弃) 并使用dup2() 使所有这些描述符指向这个设备。
  7. 核心业务逻辑

守护进程案例

写一个守护进程,每隔2s获取一下系统时间,将这个时间写入到磁盘文件中。

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

void work(int num) {
    // 捕捉到信号之后,获取系统时间,写入磁盘文件
    time_t tm = time(NULL);
    struct tm * loc = localtime(&tm);
    // char buf[1024];

    // sprintf(buf, "%d-%d-%d %d:%d:%d\n",loc->tm_year,loc->tm_mon
    // ,loc->tm_mday, loc->tm_hour, loc->tm_min, loc->tm_sec);

    // printf("%s\n", buf);

    char * str = asctime(loc);
    int fd = open("time.txt", O_RDWR | O_CREAT | O_APPEND, 0664);
    write(fd ,str, strlen(str));
    close(fd);
}

int main() {

    // 1.创建子进程,退出父进程
    pid_t pid = fork();

    if(pid > 0) {
        exit(0);
    }

    // 2.将子进程重新创建一个会话
    setsid();

    // 3.设置掩码
    umask(022);

    // 4.更改工作目录
    chdir("/home/nowcoder/");

    // 5. 关闭、重定向文件描述符
    int fd = open("/dev/null", O_RDWR);
    dup2(fd, STDIN_FILENO);
    dup2(fd, STDOUT_FILENO);
    dup2(fd, STDERR_FILENO);

    // 6.业务逻辑

    // 捕捉定时信号
    struct sigaction act;
    act.sa_flags = 0;
    act.sa_handler = work;
    sigemptyset(&act.sa_mask);
    sigaction(SIGALRM, &act, NULL);

    struct itimerval val;
    val.it_value.tv_sec = 2;
    val.it_value.tv_usec = 0;
    val.it_interval.tv_sec = 2;
    val.it_interval.tv_usec = 0;

    // 创建定时器
    setitimer(ITIMER_REAL, &val, NULL);

    // 不让进程结束
    while(1) {
        sleep(10);
    }

    return 0;
}