特殊进程之守护进程

文章目录

  • 1、守护进程的概念
  • 2、如何查看守护进程
  • 3、编写守护进程的步骤
    • 3.1 创建子进程,父进程退出
    • 3.2 在子进程中创建新会话
    • 3.3 改变当前工作目录
    • 3.4 重设文件权限掩码
    • 3.5 关闭不需要的文件描述符
    • 3.6 某些特殊的守护进程打开/dev/null
  • 4、守护进程代码示例

1、守护进程的概念

守护进程(Daemon Process),也就是通常说的 Daemon 进程(精灵进程),是 Linux 中的后台服务进程。它是一个生存期较长的进程,通常独立于控制终端并且周期性地执行某种任务或等待处理某些发生的事件。其他进程都是在用户登录或运行程序时创建,在运行结束或者用户注销时终止,但系统服务进程不受用户登录注销的影响,它们一直运行,这就是守护进程。

守护进程是个特殊的孤儿进程,这种进程脱离终端,为什么要脱离终端呢?之所以脱离于终端是为了避免进程被任何终端所产生的信息所打断,其在执行过程中的信息也不在任何终端上显示。由于在 Linux 中,每一个系统与用户进行交流的界面称为终端,每一个从此终端开始运行的进程都会依附于这个终端,这个终端就称为这些进程的控制终端,当控制终端被关闭时,相应的进程都会自动关闭。

Linux 的大多数服务器就是用守护进程实现的。比如,Internet 服务器 inetd,Web 服务器 httpd 等。

2、如何查看守护进程

在终端上使用命令 ps axj

a:表示显示所有进程,包括其他用户的进程。
x:不仅可以显示有控制终端的进程,也可以显示没有控制终端的进程。
j:表示列出与作业控制相关的信息。

在这里插入图片描述

上述字段含义如下:
PPID:父进程ID。

PID:当前进程ID。

PGID:当前进程的进程组ID。

SID:会话ID。

TTY:该进程在哪个终端下运作,其中“?”表示与终端机无关,例如守护进程;tty1-tty6是本机上的登录者进程;pts/0等表示网络连接进主机的进程。

TPGID:终端进程组ID。

STAT:进程状态,其中“S”表示睡眠状态,“R”表示运行状态,“Z”表示僵尸状态,“T”表示停止状态,“W”表示等待状态。

UID:用户ID。

TIME:该进程使用的CPU时间。

COMMAND:正在运行的进程的命令名。

从上图中可以看出守护进程都有以下特点:

守护进程基本上都是以超级用户启动( UID 为 0 )
没有控制终端( TTY 为 ?)
终端进程组 ID 为 -1 ( TPGID 表示终端进程组 ID)

注意:COMMAND字段带有[ ]的叫内核守护进程,不带[ ]的叫普通守护进程,也叫做用户守护进程。

一般情况下,守护进程可以通过以下方式启动:

  1. 在系统启动时由启动脚本启动,这些启动脚本通常放在 /etc/rc.d 目录下
  2. 利用 inetd 超级服务器启动,如 telnet 等
  3. 由 cron 定时启动以及在终端用 nohup 启动的进程也是守护进程

这里面存放的基本都是守护进程的脚本
在这里插入图片描述

在Linux中,守护进程有两种方式,一种是svsy方式,一种是xinetd方式(超级守护进程)。 每个守护进程都会有一个脚本,可以理解成工作配置文件,守护进程的脚本需要放在指定位置,独立启动守护进程:放在/etc/rc.d 目录下,当然也包括xinet的shell脚本;超级守护进程:按照xinet中脚本的指示,它所管理的守护进程位于/etc/xinetd.config目录下。

sysv:

独立启动,一开机运行就会进入内存,一直处于listen状态,即使该守护进程不运行也会一直占用系统资源,但是其最大的优点就是,它一直启动,当有请求时会立即响应,响应速度快,比如http服务,这样的进程都保存在/etc/rc.d/init.d目录下

xinet d:

超级守护进程,管理众多的进程,比如telnet服务。xinetd自己是一个sysv,它就像老板一样,自己常驻于内存,管理其它的进程,其它进程就相当于它的员工,在其它进程没有用时会睡眠,并不占用系统资源,当有工作时候老板xinetd会通知它的员工,唤醒某个进程来执行作业。这种方式适合于那些不是经常被人使用,不需要常驻内存的程序,但是此方式响应时间长,但是节省系统资源,方便管理。超级守护进程的配置文件是/etc/xinetd.conf,超级守护进程的子进程们存放在/etc/xinetd.d/目录下

3、编写守护进程的步骤

3.1 创建子进程,父进程退出

由于守护进程是脱离控制终端的,因此完成第一步后子进程变成后台进程。之后的所有工作都在子进程中完成。而用户通过 shell 可以执行其他的命令,从而在形式上做到了与控制终端的脱离。

虽然父进程退出了,但是子进程也不是进程组的组长进程,因为父进程退出,子进程成为孤儿进程,接着子进程会被init进程给领养,成为init 进程的子进程

父进程先退出,子进程就会成为孤儿进程
子进程退出,父进程没有进行wait,子进程会成为僵尸进程

3.2 在子进程中创建新会话

这个步骤是创建守护进程中最重要的一步,在这里使用的函数是 setsid() 。

这里先要明确两个概念:进程组和会话期。

进程组

进程组是一个或多个进程的集合。进程组由进程组 ID 来唯一标识。除了进程号( PID )之外,进程组 ID 也是一个进程的必备属性。
每个进程组都有一个组长进程,其组长进程的进程号等于进程组 ID ,且进程组 ID 不会因组长进程的退出而受到影响。

会话期

会话期是一个或多个进程组的集合。通常一个会话开始于用户登录,终止于用户退出;或者说开始于终端打开,结束于终端关闭。会话期的第一个进程称为会话组长。在此期间该用户运行的所有进程都属于这个会话期。

进程组和会话期之间的关系如图:

在这里插入图片描述

setsid()函数说明
使用指令 man 2 setsid 查看详细信息

#include <sys/types.h>
#include <unistd.h>pid_t setsid(void);

功能:
  如果调用进程不是进程组长,则 setsid() 将创建一个新会话。调用进程将成为新会话的会话组组长(即,其会话 ID 与其进程 ID 相同)。同时调用进程也将成为会话中新进程组的进程组组长(即,其进程组 ID 与其进程 ID 相同)。调用进程将是新进程组和新会话中的唯一进程。
参数:无
返回:
  成功:返回调用进程的(新)会话ID
  失败:返回(pid_t)-1,并设置 errno

上面已经提到,setsid() 函数用于创建一个新的会话,并担任该会话的组长,所以调用 setsid() 有下面 3 个作用

1、让进程摆脱原会话的控制
2、让进程摆脱原进程组的控制
3、让进程摆脱原控制终端的控制

由于在调用 fork() 函数时,子进程 全盘复制 了父进程的会话期进程组和控制终端等。所以虽然父进程退出了,但原先的 会话期、进程组、控制终端等并没有改变,因此,子进程并不是真正意义上的独立,而 setsid() 函数能够使进程完全独立出来,从而脱离所有其他进程的控制。

3.3 改变当前工作目录

使用 fork() 函数创建的子进程是完全继承了父进程的当前工作目录,所以从父进程继承过来的当前工作目录可能是一个挂载的文件系统中。因为守护进程有一般情况是在系统在引导之前是一直从在的,所以在进程工作的过程中当前目录所在的文件系统(比如“/mnt/usb” 等)是不能卸载的。

因此,一般的做法是将根目录作为守护进程的当前工作目录,这样就可以避免上述问题。当然,如有特殊需要,也可以把当前工作目录换成其他的路径,如“/tmp”。

改变工作目录的函数是 chdir() 函数,其函数原型如下所示:

#include <unistd.h>int chdir(const char *path);

功能:
  改变调用者的工作目录
参数:
  path:新的工作目录的路径
返回:
  成功:返回0
  失败:返回-1,同时设置errno

3.4 重设文件权限掩码

文件权限掩码(通常用八进制表示)的作用是屏蔽文件权限中的对应位。例如,如果文件权限掩码是0050,它表示屏蔽了文件所属用户组的可读与可执行权限。由于使用 fork() 函数新建的子进程继承了父进程的文件权限掩码,这就给该子进程使用文件带来了一定的影响。如果守护进程需要创建文件,那么他可能需要设置特定的权限。因此,把文件权限掩码设置为一个已知的值(通常设置为0),可以增强该守护进程的灵活性。

umask的数值共有四位,例如上面的输出0050,四位数表示四组权限值,分别是文件特殊权限,文件所有者权限,文件所属用户组权限,其他用户权限。

这里我们先忽略掉文件特殊权限位。

可读权限r表示4,可写权限w表示2,可执行权限x表示1

umask值指的是需要从原始默认权限减掉的权限!我们已经知道r、w、x的数值分别是4、2、1。 所以如果要去掉可读和可执行权限,umask值中相应的位就是5

如果要去掉读权限,那就是4,去掉读与写权限,就是6,去掉执行与写权限,就是3,去掉写的权限,就是5!

新建文件和目录的默认权限值就是在原始默认权限的基础上去掉umask值,umask值与原始默认权限共同决定了新建文件和目录的默认权限值。

在使用open()建立新文件时, 该参数mode 并非真正建立文件的权限, 而是 (mode&~umask)的权限值。

设置文件权限掩码的函数是 umask()。在这里,通常的使用方法为 umask(0)。其函数原型如下所示:

#include <sys/types.h>
#include <sys/stat.h>mode_t umask(mode_t mask);

功能:
  umask() 将调用进程的文件模式创建掩码(umask)设置为 mask & 0777(即仅使用掩码的文件权限位)。
参数:
  mask:要设置的权限值,用八进制表示
返回:
  此系统调用始终成功,并返回掩码的上一个值。

3.5 关闭不需要的文件描述符

同样地,用 fork() 函数新建的子进程会从父进程那里继承一些已经打开了的文件。这些被打开的文件可能永远不会被守护进程访问,但它们一样占用系统资源,而且还可能导致所在的文件系统无法被卸载。

特别是守护进程和终端无关,所以指向终端设备的标准输入、标准输出和标准错误流等已经不再使用,应当被关闭。

可以使用函数 getdtablesize() 来获取当前进程文件描述符表的大小,并通过使用 close() 来依次关闭。

函数原型如下:

#include <unistd.h>
int getdtablesize(void);

getdtablesize()函数返回进程可以打开的最大文件数,比文件描述符的最大可能值多一个。

#include <unistd.h>
int close(int fd);

close() 用于关闭文件描述符,关闭成功则返回 0,失败则返回 -1 并设置 errno

所以关闭文件描述符的代码可以如下写法:

int num = getdtablesize(); // 获取当前进程文件描述符表大小for (int i = 0; i < num; i++)  
{close (i);
}

3.6 某些特殊的守护进程打开/dev/null

某些特殊的守护进程打开/dev/null,使其具有文件描述符0、1、2,这样任何一个试图读标准输入、标准输出、标准出错时都不会有任何效果,这样符合了守护进程不与终端设备相关联的属性。

/dev/null 是Linux下的黑洞文件,向里面写入的所有数据都将被忽略

4、守护进程代码示例

#include <stdio.h>     //for perror...
#include <string.h>    //for strlen...
#include <stdlib.h>    //for EXIT_FAILURE EXIT_SUCCESS...
#include <fcntl.h>     //for O_RDWR | O_CREAT | O_APPEND...
#include <unistd.h>    //for fork chidr setsid getdtablesize close...
#include <sys/types.h> //for umask...
#include <signal.h>    //for signal...volatile sig_atomic_t runing = 1;void sigint_handler(int sig)
{int fd = open("/tmp/dameon.log2", O_RDWR | O_CREAT | O_APPEND, 0644);char *p = "守护进程运行结束!\n";write(fd, p, strlen(p));close(fd);runing = 0;
}int main()
{// 创建子进程,父进程退出pid_t id = fork();if (id == -1){perror("fork");exit(EXIT_FAILURE);}if (id > 0) // 父进程{printf("父进程id:%d\n", getpid());exit(EXIT_SUCCESS);}//打印子进程号printf("子进程id:%d\n", getpid());// 在子进程中创建新会话pid_t temp_pid = setsid();// 改变当前的工作路径chdir("/");// 改变进程本身的umaskumask(0);int num = getdtablesize(); /* 获取当前进程文件描述符表大小 */int i = 0;for (i = 0; i < num; i++){close(i);}// 屏蔽一些控制终端操作的信号signal(SIGTTOU, SIG_IGN);signal(SIGTTIN, SIG_IGN);signal(SIGTSTP, SIG_IGN);signal(SIGHUP, SIG_IGN);// 对SIGINT进行捕获signal(SIGINT, sigint_handler);while (runing){int fd = open("/tmp/dameon.log", O_RDWR | O_CREAT | O_APPEND, 0644);if (fd == -1){perror("open");exit(EXIT_FAILURE);}char *p = "这个一个守护进程!\n";write(fd, p, strlen(p));close(fd);sleep(3);}return 0;
}

编译然后运行

在这里插入图片描述

查看对应的日志文件

在这里插入图片描述

向这个进程发送2号信号,进程则会捕获到2号信号,触发自定义函数,再次查看进程,发现进程已经结束

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/259885.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

3接上篇 我的自定义GPTs的改进优化 与物理世界连接成功 GPTs的创建与使用定义和执行特定任务的功能模块 通过API与外部系统或服务的交互

https://blog.csdn.net/chenhao0568/article/details/134875067?spm1001.2014.3001.5502 从服务器日志里看到请求多了一个“location” 23.102.140.123 - - [08/Dec/2023:14:02:20 0800] "GET /getWeather.php?location&locationNewYork HTTP/1.1" 200 337 &…

深度学习还可以从如下方面进行创新!!

文章目录 一、我认为可以从如下5个方向进行创新总结 一、我认为可以从如下5个方向进行创新 新的模型结构&#xff1a;尽管现在的深度学习模型已经非常强大&#xff0c;但是还有很多未被探索的模型结构。探索新的模型结构可以带来更好的性能和更低的计算成本。 新的优化算法&a…

【力扣热题100】287. 寻找重复数(弗洛伊德的乌龟和兔子方法)

【力扣热题100】287. 寻找重复数 写在最前面理解解决 "寻找重复数" 问题的算法问题描述弗洛伊德的乌龟和兔子方法为什么这个方法有效&#xff1f; 代码复杂度 总结回顾 写在最前面 刷一道力扣热题100吧 难度中等 https://leetcode.cn/problems/find-the-duplicate-…

让聪明的车连接智慧的路,C-V2X开启智慧出行生活

“聪明的车 智慧的路”形容的便是车路协同的智慧交通系统&#xff0c;从具备无钥匙启动&#xff0c;智能辅助驾驶和丰富娱乐影音功能的智能网联汽车&#xff0c;到园区的无人快递配送车&#xff0c;和开放的城市道路上自动驾驶的公交车、出租车&#xff0c;越来越多的车联网应用…

Linux——web网站服务(一)

一、安装httpd服务器Apache网站服务 1、准备工作 为了避免发送端口冲突&#xff0c;程序冲突等现象&#xff0c;卸载使用rpm方式安装的httpd #使用命令检查是否下载了httpd [rootserver ~]# rpm -qa httpd #如果有则使用 [rootserver ~]# rpm -e httpd --nodeps Apache的配置…

SDXL使用animateDiff和hotshot-xl进行文生视频

截至2023.12.8号&#xff0c;目前市面上有两款适用于SDXL的文生视频开源工具&#xff0c;分别是AnimateDiff和hotshot-xl。 一、工具下载链接 &#xff08;1&#xff09;AnimateDiff的webui版本的git链接&#xff1a; GitHub - continue-revolution/sd-webui-animatediff: A…

Java数据结构06——树

1.why: 数组&链表&树 2. 大纲 2.1前中后序 public class HeroNode {private int no;private String name;private HeroNode left;//默认为nullprivate HeroNode right;//默认为nullpublic HeroNode(int no, String name) {this.no no;this.name name;}public int …

C/C++端口复用SO_REUSEADDR(setsockopt参数),test ok

端口复用最常用的用途应该是防止服务器重启时之前绑定的端口还未释放或者程序突然退出而系统没有释放端口。这种情况下如果设定了端口复用&#xff0c;则新启动的服务器进程可以直接绑定端口。如果没有设定端口复用&#xff0c;绑定会失败&#xff0c;提示ADDR已经在使用中——…

Navicat 技术指引 | 适用于 GaussDB 分布式的日志查询与配置设置

Navicat Premium&#xff08;16.3.3 Windows 版或以上&#xff09;正式支持 GaussDB 分布式数据库。GaussDB 分布式模式更适合对系统可用性和数据处理能力要求较高的场景。Navicat 工具不仅提供可视化数据查看和编辑功能&#xff0c;还提供强大的高阶功能&#xff08;如模型、结…

[笔记]ARMv7/ARMv8 交叉编译器下载

开发 Cortex-A7、Cortex-A72 或其他 ARM 架构 profile 芯片时&#xff0c;经常需要下载对应架构的交叉编译器&#xff0c;所以写这篇笔记&#xff0c;用于记录一下交叉编译器下载流程&#xff0c;免得搞忘。 编译环境&#xff1a;ubuntu 虚拟机 下载地址 我们可以从 ARM 官网…

深入浅出:HTTPS单向与双向认证及证书解析20231208

介绍: 网络安全的核心之一是了解和实施HTTPS认证。本文将探讨HTTPS单向认证和双向认证的区别&#xff0c;以及SSL证书和CA证书在这些过程中的作用&#xff0c;并通过Nginx配置实例具体说明。 第一部分&#xff1a;HTTPS单向认证 定义及工作原理&#xff1a;HTTPS单向认证是一…

C语言WFC实现绘制贝塞尔曲线的函数

前言&#xff1a; 贝塞尔曲线于 1962 年&#xff0c;由法国工程师皮埃尔贝济埃&#xff08;Pierre Bzier&#xff09;所广泛发表&#xff0c;他运用贝塞尔曲线来为汽车的主体进行设计,贝塞尔曲线最初由保尔德卡斯特里奥于1959年运用德卡斯特里奥算法开发&#xff0c;以稳定数值…