特殊进程之守护进程

文章目录

  • 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.rhkb.cn/news/213442.html

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

相关文章

一对一聊天程序

package untitled1.src;import javax.swing.*; import java.awt.*; import java.awt.event.ActionEvent; import java.awt.event.ActionListener; import java.io.*; import java.net.*;public class MyServer extends JFrame{private ServerSocket server; // 服务器套接字pri…

22、pytest多个参数化的组合

官方实例 # content of test_multi_parametrie.py import pytestpytest.mark.parametrize("x",[0,1]) pytest.mark.parametrize("y",[2,3]) def test_foo(x,y):print("{}-{}".format(x,y))pass解读与实操 要获得多个参数化参数的所有组合&…

解决使用pnpm安装时Sharp模块报错的方法

在使用pnpm进行项目依赖安装的过程中&#xff0c;有时候会遇到Sharp模块报错的情况。Sharp是一个用于处理图像的Node.js模块&#xff0c;但它的安装可能会因为各种原因而失败&#xff0c;导致项目无法正常启动。本文将介绍这个问题的方法。 问题描述 解决方法 在命令行分别输…

jsp在线辅助教育系统Myeclipse开发mysql数据库web结构java编程计算机网页项目

一、源码特点 JSP 在线辅助教育系统是一套完善的java web信息管理系统&#xff0c;对理解JSP java编程开发语言有帮助&#xff0c;系统具有完整的源代码和数据库&#xff0c;系统主要采用B/S模式开发。开发环境为TOMCAT7.0,Myeclipse8.5开发&#xff0c;数据库为Mysql5.0&…

概率密度函数(PDF)正态分布

概率密度函数&#xff08;PDF&#xff09;是一个描述连续随机变量取特定值的相对可能性的函数。对于正态分布的情况&#xff0c;其PDF有一个特定的形式&#xff0c;这个形式中包括了一个常数乘以一个指数函数&#xff0c;它假设误差项服从均值为0的正态分布&#xff1a; p ( …

【数据结构】——队列实现二叉树的功能

前言&#xff1a;二叉树的实现方式多种多样&#xff0c;有数组实现满二叉树&#xff0c;有链表实现完全二叉树&#xff0c;今天我们就用队列来实现二叉树。 创建二叉树&#xff1a; typedef int BTDataType; typedef struct BinaryTreeNode {BTDataType data;struct BinaryTre…

java8 常用code

文章目录 前言一、lambda1. 排序1.1 按照对象属性排序&#xff1a;1.2 字符串List排序&#xff1a;1.3 数据库排序jpa 2. 聚合2.1 基本聚合&#xff08;返回对象list&#xff09;2.2 多字段组合聚合&#xff08;直接返回对象list数量&#xff09; 二、基础语法2.1 List2.1.1 数…

Java对象转Map

在和外部系统对接时&#xff0c;对方系统提供的SDK方法入参全是Map&#xff0c;没办法&#xff0c;只能想办法把对象转成Map。这里&#xff0c;借助了hutool的工具类&#xff0c;可以方便的通过反射获取对象的属性。引入hutool的maven配置&#xff1a; <dependency><g…

Flink 使用场景

Apache Flink 功能强大&#xff0c;支持开发和运行多种不同种类的应用程序。它的主要特性包括&#xff1a;批流一体化、精密的状态管理、事件时间支持以及精确一次的状态一致性保障等。Flink 不仅可以运行在包括 YARN、 Mesos、K8s 在内的多种资源管理框架上&#xff0c;还支持…

智慧社区前景无限,科技引领未来发展

社区是城镇化发展的标志&#xff0c;作为人类现代社会的生活的基本圈子&#xff0c;是人类生活离不开的地方&#xff0c;社区人口密度大、车辆多&#xff0c;管理无序&#xff0c;社区的膨胀式发展多多少少带来一定的管理上的缺失。社区作为智慧城市建设的重要一环&#xff0c;…

时间复杂度为 O(n^2) 的排序算法 | 京东物流技术团队

对于小规模数据&#xff0c;我们可以选用时间复杂度为 O(n2) 的排序算法。因为时间复杂度并不代表实际代码的执行时间&#xff0c;它省去了低阶、系数和常数&#xff0c;仅代表的增长趋势&#xff0c;所以在小规模数据情况下&#xff0c; O(n2) 的排序算法可能会比 O(nlogn) 的…

uniapp实战 —— 竖排多级分类展示

效果预览 完整范例代码 页面 src\pages\category\category.vue <script setup lang"ts"> import { getCategoryTopAPI } from /apis/category import type { CategoryTopItem } from /types/category import { onLoad } from dcloudio/uni-app import { compu…

【链表Linked List】力扣-114 二叉树展开为链表

目录 题目描述 解题过程 官方题解 题目描述 给你二叉树的根结点 root &#xff0c;请你将它展开为一个单链表&#xff1a; 展开后的单链表应该同样使用 TreeNode &#xff0c;其中 right 子指针指向链表中下一个结点&#xff0c;而左子指针始终为 null 。展开后的单链表应…

【Vulnhub 靶场】【Momentum: 2】【简单】【20210628】

1、环境介绍 靶场介绍&#xff1a;https://www.vulnhub.com/entry/momentum-2,702/ 靶场下载&#xff1a;https://download.vulnhub.com/momentum/Momentum2.ova 靶场难度&#xff1a;简单 发布日期&#xff1a;2021年06月28日 文件大小&#xff1a;698 MB 靶场作者&#xff1…

在OpenCV基于深度学习的超分辨率模型实践

1. 引言 OpenCV是一个开源的计算机视觉库&#xff0c;拥有大量优秀的算法。基于最新的合并&#xff0c;OpenCV包含一个易于使用的接口&#xff0c;主要用于实现基于深度学习方法的超分辨率&#xff08;SR&#xff09;。该接口包含预先训练的模型&#xff0c;这些模型可以非常容…

如何为 3D 模型制作纹理的最佳方法

在线工具推荐&#xff1a; 3D数字孪生场景编辑器 - GLTF/GLB材质纹理编辑器 - 3D模型在线转换 - Three.js AI自动纹理开发包 - YOLO 虚幻合成数据生成器 - 三维模型预览图生成器 - 3D模型语义搜索引擎 您可以通过不同的方式为 3D 模型创建 3D 纹理。下面我们将介绍为 3D …

小调查:你申请的流量卡,快递员派件时让你激活并充话费了吗?

说到这个问题&#xff0c;就要给大家普及一下流量卡的激活方式了&#xff0c;并不是所有的流量卡快递都需要快递激活并充话费&#xff0c;只有在套餐详情种明确标注快递激活的流量卡才会有这个要求&#xff0c;自主激活的流量卡则不需要的。 如图所示&#xff1a; 接下来&#…

【征稿倒计时十天】第三届高性能计算与通信工程国际学术会议(HPCCE 2023)

【有ISSN、ISBN号&#xff01;&#xff01;往届均已完成EI检索】 第三届高性能计算与通信工程国际学术会议(HPCCE 2023) 2023 3rd International Conference on High Performance Computing and Communication Engineering (HPCCE 2023) 2023年12月22-24日 | 中国哈尔滨 第三…

听GPT 讲Rust源代码--src/tools(9)

File: rust/src/tools/rust-analyzer/crates/ide-assists/src/handlers/apply_demorgan.rs 在Rust源代码中&#xff0c;apply_demorgan.rs文件位于rust-analyzer工具的ide-assists库中&#xff0c;其作用是实现一个辅助函数&#xff0c;用于在代码中应用De Morgan定律的变换。 …

空间运算设备-Apple Vision Pro

苹果以其在科技领域的创新而闻名&#xff0c;他们致力于推动技术的边界&#xff0c;这在他们的产品中表现得非常明显。他们尝试开发一项的新型突破性显示技术。在 2023 年 6 月 5 日官网宣布将发布 Apple Vision Pro 头戴空间设备&#xff0c;我们一起来了解一下 Apple Vision …