Linux学习笔记之五(父子进程、孤儿进程、僵尸进程、守护进程)

Linux

  • 1、进程
    • 1.1、进程的六种状态
    • 1.2、创建子进程
    • 1.3、添加子进程任务
    • 1.4、孤儿进程、僵尸进程、守护进程
      • 1.4.1、避免僵尸进程
      • 1.4.2、创建守护进程
      • 1.4.3、杀死守护进程
    • 1.5、综合练习

1、进程

进程可以简单的理解为一个正在执行的程序,它是计算机系统中拥有资源和独立运行的最小单位。多个进程同时运行从宏观看是并行,从微观上看是串行。举个例子,现有一个CPU以及两个同时运行的线程a和b,CPU实际上是用极小的时间碎片来交替执行a和b,以达到肉眼觉得CPU在同时执行两个进程的效果。
在这里插入图片描述
进程有三个状态,分别是就绪态、运行态、阻塞态。顾名思义,就绪态就是进程万事俱备只等CPU来执行它了;运行态便是CPU正在执行该进程;阻塞态是线程还没准备好被CPU执行。当然在这三个状态之上又衍生出许多状态,这里不多做介绍。
在这里插入图片描述
另外,每一个进程都有自己的编号,称之为pid(process identity document)。在进程中可以通过getpid()获得当前进程pid,也可以通过getppid()获得当前进程父进程的pid。

1.1、进程的六种状态

  1. 运行状态(Running: R):进程在运行,或者进程准备好被系统调度。
  2. 睡眠状态(Sleeping: S):此时进程在安静的等待某个事件发生,且此时进程也可以被杀死。
  3. 磁盘休眠状态(Disk sleep: D):不可杀死的睡眠状态。
  4. 停止状态(Stopped: T):该进程被某个信号叫停了,同时它也可以被叫起来继续运行。
  5. 僵尸状态(Zombie: Z):子进程死了,但父进程在忙无法替它收尸,此时子进程进入僵尸状态。
  6. 死亡状态(Dead: X):进程被杀死之后,尸体也成功回收,即资源被回收。

1.2、创建子进程

man 2 fork

在这里插入图片描述
可以看到通过以下代码便可以创建一个子进程。

pid_t pid fork();

返回值:成功则返回子进程的pid,失败则返回负值。

用fork创建的子进程会和父进程执行同一个可执行文件,但子进程会从fork函数之后才开始执行。如图所示:
在这里插入图片描述
这里值得注意的是,程序的编译会经历四个步骤,即预处理、编译、汇编、链接。只有经过这四个步骤之后程序才会变成一个可执行文件,而由于这四个步骤会处理好程序的各种变量、头文件、宏定义等内容,所以不会导致子进程从fork开始执行下去会因为缺少一些变量定义之类的而产生报错。

1.3、添加子进程任务

如果仅仅使用fork让子进程执行父进程的代码,这将使子进程显得毫无意义,而为了给子进程添加新的任务,exec函数族便被发明出来。从说明书可以看到exec有六个函数。

man execl

在这里插入图片描述
比较常用的使execl,通过用execl函数让子进程去执行其他的可执行文件,以达到给子进程添加新任务的目的。其函数原型长这样:

int execl(const char *path, const char *arg, ... /* (char *) NULL*/);

它的参数应当如何设置,我先直接贴一段manual的原文上来。

The  const  char *arg and subsequent ellipses in the execl(), execlp(),and execle() functions can be thought of  as  arg0,  arg1,  ...,  argn.
Together  they  describe  a list of one or more pointers to null-terminated strings that represent the argument list available  to  the  executed  program.  
The first argument, by convention, should point to the filename associated with the file being executed.  
The  list  of  arguments  must be terminated by a null pointer, and, since these are variadic functions, this pointer must be cast (char *) NULL.

这段话大概的意思是,execl可以有无数个参数,具体取决了即将调用的可执行文件的需要。但除了char *path之外的第一个参数是可执行文件的名字,最后一个参数是NULL。

char *path:可执行文件所在的目录(包含可执行文件的名字)。
char *arg1:可执行文件的名字。

char *argn:NULL

比如我们想在子进程中执行ls。execl可以这个写:

execl(/bin/ls”,“ls”,NULL);		//仅列出当前目录可见文件
execl(/bin/ls”,“ls”, "-l", NULL);		//列出当前目录可见文件详细信息
execl(/bin/ls”,“ls”, "-i", "-l", NULL);		//显示文件的inode信息
...

在这里插入图片描述

为什么父进程的getpid()和子进程getppid()得到的值不一样,可以参考下面这篇文章:父进程中getpid()值与子进程中getppid()值不相同的问题及解释

1.4、孤儿进程、僵尸进程、守护进程

  • 孤儿进程(Orphan process):父进程已经结束,子进程还在继续执行。但由于子进程需要父进程来帮助其释放资源,所以孤儿进程会被托管在 i n i t init init进程之下。
  • 僵尸进程(Zombie process):子进程已经结束,但父进程还在运行,且此时父进程无法去帮助子进程释放资源,即父进程没有读到子进程的exit()函数。导致子进程死了无人收尸,故被称之为僵尸进程。
  • 守护进程(Daemon process):一些脱离于终端,且不与用户交互的后台进程叫做守护进程。它们的存在至关重要,在背后维护着系统或某个软件、程序正常运行。下图红框内就基本是Windows系统的守护进程。
    在这里插入图片描述

1.4.1、避免僵尸进程

以上这三种进程中,孤儿进程是可以成为进入守护进程的前提,而守护进程又在许多情况下挥发巨大的作业,那么只剩下僵尸进程是程序不愿意看到的。
避免僵尸进程可以用wait系列函数函数,我们来看看它的函数说明:

man 2 wait

在这里插入图片描述
使用wait()函数得先理清一些概念:

  1. 父进程一般不执行其他任务,它的存在就是为了生出一堆子进程,再派子进程去执行具体任务,然后如果子进程死亡了,父进程再去替它们收尸。
  2. 子进程死亡之后会向父进程发送一个SIGCHILD信号,请求父进程为自己收尸(回收资源),以避免自己成为僵尸进程。
  3. wait()函数的出现就完美的满足父进程只生娃、收尸、不干事的需求,当父进程调用这个函数之后就进入阻塞状态,只有在子进程死后发送SIGCHILD信号,父进程才会醒过来去收拾子进程的资源。

wait的函数原型是:

pid_t wait(int *status);

当调用wait()函数,父进程会自动检查子进程的状态,无需我们再干预。

int *status:是一个32位的整形数据,其中包含了退出码、终止信号等信息。通常通过一些宏函数来读取status中的具体信息。当然,如果你压根不想要读取这些信息,只想默默收尸走人,那这个参数可以是NULL。
返回值:如果成功,则返回子进程的pid,反之返回-1。

读取status的宏:

  • WEXITSTATUS:在进程正常退出的情况下读取status中的退出码并将其返回。如果退出码是负数,则用255去加这个负数。(退出码即exit(code)中的code)
  • WIFEXITED:通过解析status判断进程是否为正常退出,若是则返回1,否则返回0。

此外,还有waitpid,waitid等函数,前者常用于等待回收某个具体的子进程,后者我也不太懂了。。。
贴一段代码来看看wait怎么用:

#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>int main()
{       pid_t pid = fork();     //create a child processif(pid){       int status;wait(&status);		//waiting for the child to terminate and recliam its resourcesif(WIFEXITED(status)){       printf("The exit code is: %d\n", WEXITSTATUS(status));}}else{       printf("This is the child process.\n");sleep(2);exit(3);}return 0;
}

1.4.2、创建守护进程

前文提到,守护进程是托管在 i n i t init init下的子进程,且脱离控制终端独立运行于后台。由此引出创建一个守护进程的两个必要步骤:

  1. 使用fork()创建一个新的进程,然后在父进程中使用exit()退出。该步骤可以让子进程变成孤儿进程,进而被init进程托管。
  2. 在子进程中使用setsid()函数。该函数可以让子进程脱离原来的进程组和会话,进入一个全新的会话中去。这有这样,该进程才能脱离原来的控制终端。

这两个步骤使创建一个守护进程的必要步骤,再次也先暂停下来解释何为进程组和会话。
所谓进程组,顾名思义就是许多个进程组成的一个小组,该小组的id(Group Identity Document: GID)就是小组组长的pid。接着,会话中又会聚集了许多个小组,同理,会话id(Session Identity Dccument: SID)便是作为翘楚的进程组id(GID)。一般而言,一个会话使用一个控制终端,不过也有特殊,比如对于为守护进程所创建的新会话,我们不希望它有一个控制终端。
注:控制终端就是我们敲命令行的那个窗口,也称终端或终端窗口。在Ubuntu中直接叫terminal(终端),一个terminal对应一个shell进程。而shell是一个解释器,为终端和系统之间的交互提供桥梁。参考:link
在这里插入图片描述
接下来,添加几个步骤让讲守护进程的更具备撸棒性(robust)。

  1. 通过chdir()把当前的工作目录改成根目录。
  2. 重设文件掩码(umask),一般设为0。
  3. 关闭文件描述符,由于文件描述符是内核空间返回给应用层的一个文件“代号”,然而在守护进程中我们并不希望再与应用层产生联系,所以关闭文件描述符可以节省资源。
  4. 在子进程中再套一个进程,防止会话建立新的控制终端。

最后,就可以在守护进程中添加我们需要执行的代码了。

1.4.3、杀死守护进程

守护进程一般生命周期比较长,由于其脱离了控制终端,所以想要关闭守护进程只能等到系统完全关闭或者手动杀死它。比如用kill:

kill -9 [the pid of the daemon process]

1.5、综合练习

本次练习任务:

  1. 创建一个父进程和一个子进程,并分别打印这两个进程的pid。
  2. 在父进程离开后,打印此时托管子进程的进程的pid。
  3. 创建一个守护进程,要求更改其目录,关闭文件描述符号,修改文件掩码。
  4. 最后从控制终端杀死这个守护进程。
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>int main()
{pid_t pid = fork();if(pid) 								//enter the parent process{printf("the pid of the parent processs is: %d.\n",getpid());exit(1);}else    								//enter the child process{sleep(1); 						//waiting for the parent process to terminateprintf("the pid of the child processs is: %d, and parent is: %d. \n",getpid(),getppid());setsid();     		//create and enter a new sessionchdir("/");     				//change the working directoryumask(0);       				//change the umaskfor(int i=0;i<3;i++){close(i);} //close the file descriptorwhile(1){//you can put any programs you like into this field.}}return 0;
}

输出结果是:

the pid of parent process is: 2600.
the pid of child process is: 2601, and its parent is: 1420.

可以看到子进程的父进程已经和原来创建它的父进程pid不一样了,我们通过搜索看看是谁托管了这个子进程。

ps -aux | grep 1420

在这里插入图片描述
可以看到是init进程托管了这个子进程。此外,当该孤儿进程使用setsid()函数变成守护进程之后,如果再使用printf()之类的函数将失去效果。因为守护进程没有其对应的控制终端,自然无法让printf()发挥作用。

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

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

相关文章

小红书母婴博主投放技巧是什么,怎么避免无用功

如今&#xff0c;随着互联网的发展&#xff0c;母婴博主和社交媒体成为了很多妈妈们&#xff0c;获取育儿知识和建立社交圈的重要途径。今天为大家分享下小红书母婴博主投放技巧是什么&#xff0c;怎么避免无用功&#xff01; 一、优质的母婴博主在哪里 我们都知道&#xff0c;…

数据中台之数据分析

效果界面 技术方案 Notebook集成 在您的数据平台上,创建一个能够与Jupyter Notebook通讯的服务。通过Jupyter Notebook的HTTP API与Notebook实例进行交互,执行代码、获取输出等。用户界面 在数据开发/数据分析的代码框右上方,添加一个机器人样式的图标,用户点击后可以调起…

verdi如何打开时可以加载配置比如字体

打开tcl使能 找到配置字体的命令 其实其他有需要的文件配置都可以在这里找到对应的指令 存储文件 新建verdi001.tcl文件 输入想要调整的字体以及大小 verdiSetFont -font "Bitstream Vera Sans" -size "18" verdiSetFont -monoFont "Courier&q…

多篇论文介绍-摘要

论文地址https://arxiv.org/pdf/2301.10051.pdf 目录 01CIEFRNet&#xff1a;面向高速公路的抛洒物检测算法 02改进 YOLOv5 的 PDC 钻头复合片缺损识别 03 基于SimAM注意力机制的DCN-YOLOv5水下目标检测 04 基于改进YOLOv7-tiny 算法的输电线路螺栓缺销检测 ​编辑05 基于改进Y…

亚马逊鲲鹏系统能做什么

亚马逊鲲鹏系统是一款能绕过亚马逊智能检测&#xff0c;完全模拟人类真实行为&#xff0c;通过模拟真实的人流量来帮助你提升你的产品排名&#xff0c;让你的产品出现在搜索首页&#xff0c;从而快速提高你的销售业绩的营销工具&#xff01; 主要的功能有批量注册买家号、AI智能…

新生儿疝气:原因、科普和注意事项

引言&#xff1a; 新生儿疝气是一种在婴儿中相对较常见的状况&#xff0c;很多新父母可能对这一现象感到困惑和焦虑。疝气发生时&#xff0c;内腹腔的一部分可能穿过腹壁的弱点&#xff0c;导致腹部出现凸起。本文将科普新生儿疝气的原因&#xff0c;提供相关信息&#xff0c;…

计算机基础知识48

web应用程序 # Django框架是一款专门用来开发web应用的框架 # Web应用程序是一种可以通过浏览器访问的应用程序, B/S架构 案例&#xff1a;淘宝网、京东网... # 应用程序有两种模式: C/S&#xff1a;客户端/服务器端程序&#xff0c;这类程序一般独立运行 B/S&#xff1…

数据结构与算法C语言版学习笔记(5)-串,匹配算法、KMP算法

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、串的定义二、串的存储结构1.顺序结构2.链式结构 三、串的朴素的模式匹配算法&#xff08;暴力匹配算法&#xff09;1.背景2.假设我们要从下面的主串 S"…

“最强”机器学习辅助!利用自然语言让机器人更好地理解开放性世界

原创 | 文 BFT机器人 想象一下&#xff0c;你正在国外拜访朋友&#xff0c;打开他的冰箱看看有没有能够制作一顿美味早餐的食材。最初&#xff0c;冰箱里的许多物品对你来说都很陌生&#xff0c;每个物品的包装都是你不熟悉的。你开始试图理解每个物品的用途&#xff0c;并根据…

万界星空科技MES系统软件体系架构及应用

MES系统是数字化车间的核心。MES通过数字化生产过程控制&#xff0c;借助自动化和智能化技术手段&#xff0c;实现车间制造控制智能化、生产过程透明化、制造装备数控化和生产信息集成化。生产管理MES系统主要包括车间管理系统、质量管理系统、资源管理系统及数据采集和分析系统…

Ubuntu开机无法进入系统,文件根系统目录空间不足导致?

前言&#xff1a; 自己电脑上装的是Win11和Ubuntu20双系统&#xff0c;平时就是切换着用。 偶然有次&#xff0c;Ubuntu提示文件根系统目录空间不足&#xff0c;自己没在意。 结果下次开机进入Ubuntu时候&#xff0c;芭比Q了。。进不了系统 这样的事情发生很多次了&#xff0c;…

学者观察 | 数字经济中长期发展中的区块链影响力——清华大学柴跃廷

导语 区块链是一种全新的分布式基础架构与计算范式&#xff0c;既能利用非对称加密和冗余分布存储实现信息不可篡改&#xff0c;又可以利用链式数据结构实现数据信息可溯源。当前&#xff0c;区块链技术已成为全球数据交易、金融结算、国际贸易、政务民生等领域的信息基础设施…

事务(本地事务与分布式事务)

事务 1 本地事务1.1 事务的特性1.2 事务的隔离级别1.3 事务的传播属性 2 分布式事务2.1 分布式事务基础2.1.1 CAP定理2.1.2 BASE定理 2.2 分布式事务的解决方案2.2.1 两阶段提交&#xff08;2PC&#xff09;2.2.2 TCC补偿式事务2.2.3 消息事务最终一致性 1 本地事务 1.1 事务的…

module ‘torch‘ has no attribute ‘_six‘

主要问题是torchvision的问题 在122服务器上的scvi-env2环境中 import torch import torch.nn as nnimport numpy as npfrom tqdm import tqdm from torchvision.utils import save_image, make_grid # Model Hyperparametersdataset_path ./datasetscuda True DEVICE tor…

httpRequest库代码示例

python # 首先导入所需的库 library(httpRequest) # 设置主机名和端口号 proxy_host <- proxy_port <- # 使用httpRequest库的get函数下载图片 response <- httpRequest(", proxyHost proxy_host, proxyPort proxy_port) # 确保请求成功 if (response$sta…

腾讯云真的是良心云!服务器带宽、CPU、硬盘IO性能大揭秘!

本文将通过对腾讯云服务器CVM S5 4核配置的云服务器进行测试&#xff0c;来评估其在带宽、CPU和硬盘IO性能方面的表现。 在云服务器的并发处理中&#xff0c;带宽是一个重要的因素。经过测试&#xff0c;腾讯云的带宽网络表现非常出色&#xff0c;能够跑满带宽&#xff0c;同时…

表象变换与矩阵元

表象变换 一维粒子哈密顿量 表象中的矩阵元 态的表象变换 不难证明 算符的表象变换 坐标表象 Non-denumerable basis

人车实时精准管控!北斗让换流站作业更安全

换流站是高压直流输电系统的核心组成部分&#xff0c;对促进电网稳定运行、保障电力行业的可持续发展有着重要作用。长期以来&#xff0c;随着电网主变改扩建设工程的开展&#xff0c;站内作业人员安全管控压力随之增大&#xff0c;仅依靠传统的“人盯人”和“自主”管控模式较…

Java设计模式之迭代器模式

定义 提供一个对象来顺序访问聚合对象中的一系列数据&#xff0c;而不暴露聚合对象的内部表示。 结构 迭代器模式主要包含以下角色&#xff1a; 抽象聚合角色&#xff1a;定义存储、添加、删除聚合元素以及创建迭代器对象的接口。具体聚合角色&#xff1a;实现抽象聚合类&a…

mysql隐式转换转换引起的bug

生产环境中遇到一个情况情况 &#xff0c;过滤数据发现过滤不掉相关值情况&#xff0c;具体情况如下 原始数据&#xff1a; CREATE TABLE test (id bigint(11) NOT NULL AUTO_INCREMENT COMMENT 自增id,subject_id bigint(11) NOT NULL DEFAULT 0 COMMENT 主题id,subject_nam…