【Linux】守护进程(附终端、进程组、会话的介绍)

橙色

1、终端

echo $$ 

可以查看当前终端进程的id
在这里插入图片描述

  • 默认情况下(没有重定向),每个进程的标准输入、标准输出和标准错误输出都指向控制终端、进程从标准输入读也就是读用户的键盘输入,进程往标准输出或标准错误输出写也就是输出到显示器上。

2、进程组

  • 进程组和会话在进程之间形成了一种两级层次关系︰进程组是一组相关进程的集合.会话是一组相关进程组的集合。进程组合会话是为支持shell作业控制而定义的抽象概念,用户通过shell 能够交互式地在前台或后台运行命令
  • 进行组由一个或多个共享同一进程组标识符(PGID)的进程组成。一个进程组拥有一个进程组首进程,该进程是创建该组的进程,其进程ID为该进程组的 ID,新进程会继承其父进程所属的进程组 ID。
  • 进程组拥有一个生命周期.其开始时间为首进程创建组的时刻,结束时间为最后一个成员进程退出组的时刻。一个进程可能会因为终止而退出进程组。也可能会因为加入了另外一个进程组而退出进程组。进程组首进程无需是最后一个离开进程组的成员。

3、会话

  • 会话是一组进程组的集合。会话首进程是创建该新会话的进程,其进程ID会成为会话ID。新进程会继承其父进程的会话ID。
  • 一个会话中的所有进程共享单个控制终端。 控制终端会在会话首进程首次打开一个终端设备时被建立。一个终端最多可能会成为一个会话的控制终端。
  • 在任一时刻,会话中的其中一个进程组会成为终端的前台进程组,其他进程组会成为后台进程组。只有前台进程组中的进程才能从控制终端中读取输入。当用户在控制终端中输入终端字符生成信号后,该信号会被发送到前台进程组中的所有成员。
  • 当控制终端的连接建立起来之后,会话首进程会成为该终端的控制进程。

4、守护进程

  • 守护进程(Daemon Process),也就是通常说的 Daemon 进程(精灵进程),是 Linux 中的后台服务器。它是一个生存期较长的进程,通常独立于控制终端并且周期性地执行某种任务或等待某些发生的事件。一般采用 d 结尾的名字。
  • 守护进程具备以下特征:1、生命周期很长,守护进程会在系统启动的时候被创建并一直运行直至系统被关闭。2、它在后台运行并且不拥有控制终端。没有控制终端确保了内核永远不会为守护进程自动生成任何控制信号以及终端相关的信号(如SIGINT、SIGQUIT)

守护进程的创建步骤:

  • 执行一个 fork(),之后父进程退出,子进程继续执行
  • 子进程调用 setsid() 开启一个新会话
  • 清除进程的 umask 以确保当守护进程创建文件和目录时拥有所需的权限
  • 修改进程的当前工作目录,通常会改为根目录(/)。
  • 关闭守护进程从其父进程继承而来的所有打开着的文件描述符。
  • 在关闭了文件描述符 0、1、2 之后,守护进程通常会打开 /dev/null 并使用 dup2() 使所有这些描述符指向这个设备。
  • 核心业务逻辑。

代码举例:

/*
    写一个守护进程,每隔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("/root");

    // 5. 关闭、重定向文件描述符,先创建指向null的文件描述符fd,再把标准输入标准输出和标准错误重定向到null文件,
    //否则的话使用printf打印的信息会直接输出到终端。而重定向后就会输入到我定向的文件中
    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;
}

编译运行该程序
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

最后就可以看到,在我设置的工作目录(这个工作目录是可以改的,通过程序中的child即可更改)内,被成功创建了一个time.txt文件。通过vim进行浏览,可以看到时间被不停地写入到了time.txt文件内。
注意:使用ps aux可以发现./daemon是存在的,该进程是守护进程,没办法通过控制终端停止。只能使用kill -9 pid强制杀死。

猜你喜欢

转载自blog.csdn.net/mhyasadj/article/details/130849274