守护进程
终端
- 在UNIX系统中,用户通过终端登录系统后得到一个shell 进程,这个终端成为shell 进程的控制终端(Contrglling Terminal),进程中,控制终端是保存在 PCB中的信息,而fork()会复制 PCB中的信息,因此由shell 进程启动的其它进程的控制终端也是这个终端。
- 默认情况下(没有重定向),每个进程的标准输入、标准输出和标准错误输出都指向控制终端,进程从标准输入读也就是读用户的键盘输入,进程往标准输出或标准错误输出写也就是输出到显示器上。
- 在控制终端输入一些特殊的控制键可以给前台进程发信号,例如Ctrl +C会产生SIGINT信号,Ctrl +\会产生SIGQUIT信号。
# 查看当前终端的pid
echo $$
进程组
- 进程组和会话在进程之间形成了一种两级层次关系:进程组是一组相关进程的集合。会话是一组相关进程组的集合。进程组和会话是为支持 shell 作业控制而定义的抽象概念,用户通过shell能够交互式地在前台或后台运行命令。
- 进程组由一个或多个共享同一进程组标识符(PGID)的进程组成。一个进程组拥有一个进程组首进程,该进程是创建该组的进程,其进程ID为该进程组的 ID,新进程会继承其父进程所属的进程组ID。
- 进程组拥有一个生命周期。其开始时间为首进程创建组的时刻,结束时间为最后一个成员进程退出组的时刻。
- 一个进程可能会因为终止而退出进程组,也可能会因为加入了另外一个进程组而退出进程组。
- 进程组首进程无需是最后一个离开进程组的成员。
会话
- 会话是一组进程组的集合。会话首进程是创建该新会话的进程,其进程ID会成为会话ID。新进程会继承其父进程的会话ID。
- 一个会话中的所有进程共享单个控制终端。控制终端会在会话首进程首次打开一个终端设备时被建立。一个终端最多可能会成为一个会话的控制终端。
- 在任一时刻,会话中的其中一个进程组会成为终端的前台进程组,其他进程组会成为后台进程组。只有前台进程组中的进程才能从控制终端中读取输入。当用户在控制终端中输入终端字符生成信号后,该信号会被发送到前台进程组中的所有成员。
- 当控制终端的连接建立起来之后,会话首进程会成为该终端的控制进程。
进程组、会话、控制终端之间的关系
// 获取当前的进程组ID
pid_t getpgrp (void);
// 获取指定的进程的进程组ID
pid_t getpgid(pid_t pid) ;
// 设置指定的进程的进程组ID
int setpgid(pid_t pid, pid_t pgid);
// 获取指定的进程的会话ID
pid_t getsid (pid_t pid);
// 设置会话的ID
pid_t setsid ( void) ;
守护进程
- 守护进程(Daemon Process),也就是通常说的 Daemon进程(精灵进程),是Linux中的后台服务进程。它是一个生存期较长的进程,通常独立于控制终端并且周期性地执行某种任务或等待处理某些发生的事件。一般采用以 d 结尾的名字。
- 守护进程具备下列特征:
- 生命周期很长,守护进程会在系统启动的时候被创建并一直运行直至系统被关闭。
- 它在后台运行并且不拥有控制终端。没有控制终端,确保了内核永远不会为守护进程自动生成任何控制信号以及终端相关的信号(如SIGINT、SIGQUIT)。
- Linux的大多数服务器就是用守护进程实现的。比如lnternet服务器inetd,web服务器httpd等。
守护进程的创建步骤:
-
执行一个fork (),之后父进程退出,子进程继续执行。
- 进程组的首进程不能作为会话的首进程(组长进程),fork()确保该子进程不会成为进程组的首进程,从而成功调用setsid()。
- 如果父进程结束后,控制终端会显示shell提示符。为了防止控制终端显示shell提示符
-
子进程调用 setsid() 开启一个新会话。
- 目的:让新会话脱离控制终端(新会话只要不和控制终端连接就没有)
- 控制终端 vs. 终端
- 控制终端:对该进程进行操作
- 终端:该进程可以向终端输出信息。
- 控制终端 vs. 终端
- “子进程” 目的:防止建立新进程组和会话时不会和之前的进程组/会话产生冲突。
- 目的:让新会话脱离控制终端(新会话只要不和控制终端连接就没有)
-
清除进程的 umask 以确保当守护进程创建文件和目录时拥有所需的权限。
-
修改进程的当前工作目录,通常会改为根目录(/)。
-
保证守护进程在系统启动的时候被创建并一直运行直至系统被关闭。
(比如原来工作目录在U盘就很可能无法实现)
-
-
关闭守护进程从其父进程继承而来的所有打开着的文件描述符。
- 原有的文件描述符0 1 2对应的标准输入/输出/错误,有可能会把一些数据输出到原来终端上。
- 可能会一直占用某个文件(处于打开状态)导致文件无法关闭或U盘无法卸载
-
在关闭了文件描述符0、1、2之后,守护进程通常会打开/dev/null(输出到这里会默认丢弃),并使用dup2()使所有这些描述符指向这个设备。
-
核心业务逻辑
案例:写一个守护进程,每隔2s获取一下系统时间,将这个时间写入到磁盘文件中
#define _DEFAULT_SOURCE
#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.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);
// 格式化
// 1.
// 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);
// 2.
char* str = asctime(loc);
int fd = open("/home/boyangcao/Linux/Lesson28/time.txt", O_RDWR | O_CREAT | O_APPEND, 0664);
write(fd, str, strlen(str));
}
int main()
{
// 1. 创建一个子进程,退出父进程
pid_t pid = fork();
if(pid > 0) exit(0);
// 2. 将子进程重新创建一个会话
setsid();
// 3. 设置掩码
umask(022);
// 4. 更改工作目录
chdir("/");
// 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;
}