linux并发服务器 —— 多进程并发(四)
进程概述
程序是包含一系列信息的文件,描述了如何在运行时创建一个进程;
进程是正在运行的程序的实例,可以用一个程序来创建多个进程;
用户内存空间包含程序代码以及代码所使用的变量,内核数据结构用于维护进程状态信息;
进程控制块(PCB):维护进程相关的信息,task_struct结构体
PCB内部成员:进程id、进程的状态、进程切换时需要保存和恢复的一些CPU寄存器、虚拟地址空间信息、控制终端信息等
进程可以使用的资源上线可以调用: ulimit -a 进行查询
进程状态转换
三态模型:就绪、运行、阻塞
五态模型:新建、就绪、运行、阻塞、终止
阻塞态不能直接变为运行态,需要先变为就绪态;
新建态:进程刚被创建,还没有分配资源,尚未进入就绪队列;
终止态:进程完成任务到达正常结束点,或出现错误而异常终止,或被新操作系统以及有终止权的进程所终止;
查看进程:ps aux/ajx(不能动态显示)
a - 显示终端所有进程;
u - 显示进程详细信息
x - 显示没有控制终端的进程;
j - 列出与作业控制相关的信息
实时显示进程动态:top (-d 指定时间间隔)
按键排序:
M - 内存降序
P - CPU占有率降序
U - 根据用户名筛选
K - 杀死进程
T - 根据运行时长排序
杀死进程:kill PID
kill -9 PID(强制杀死进程)
killall name 根据进程名杀死进程;
进程号相关函数
进程号的范围 0~32767;
getpid(void)、getppid(void)、getpgid(pid_t pid)
进程创建
/*#include <sys/types.h>#include <unistd.h>pid_t fork(void);返回值:返回两次;一次在父进程中,一次在子进程中父进程中返回子进程的ID子进程中返回0如何区分父进程和子进程 - 通过fork返回值; 父进程中返回-1表示创建子进程失败,并设置errno失败的原因:1. 进程数上限2. 系统内存不足
*/
#include <iostream>
#include <sys/types.h>
#include <unistd.h>
using namespace std;
int main(){// 创建子进程pid_t pid = fork();// 判断父子进程if(pid>0){cout<<pid<<endl;cout<<"父进程 - 进程号:"<<getpid()<<endl;}else if(pid == 0){cout<<"子进程 - 进程号:"<<getpid()<<endl;}for(int i = 0 ; i<3 ; i++){cout<<"i: "<<i<<" "<<getpid()<<endl;sleep(1);}return 0;
}
父子进程虚拟地址空间的情况
子进程用户区数据和父进程一样,内核区也会拷贝,但pid不同;
fork()是通过写时拷贝实现的,资源的复制在需要写入时才进行,在此之前以只读方式进行共享;
父子进程的关系及GDB多线程调试
父子进程间的关系
区别:
1. fork()返回值不同
2. pcb中的一些数据 eg. 当前进程pid ppid、信号集
共同点:
子进程刚被创建,没执行任何写操作
- 用户区数据
- 文件描述符表
父子进程对变量是不是共享的?
- 读时共享,写时拷贝;
GDB多进程调试
GDB默认只能跟踪一个进程 默认跟踪父进程;
- 显示跟踪进程:show follow-fork-mode
- 设置调试父进程和子进程:set follow-fork-mode [parent(默认)|child]
- 显示调试模式:show detach-on-fork
- 设置调试模式:set detach-on-fork [on|off]
默认为on,表示调式当前进程时,其他进程继续运行;off表示调式当前进程,其它进程被GDB挂起,停在fork处;
- 查看调试的进程:info inferiors
- 切换调试进程:inferior 进程编号 后 c即可
- 使进程脱离GDB调试:detach inferior id
exec函数族(一系列函数)
作用:根据指定文件名找到可执行文件;用其取代调用进程的内容(在调用进程内部执行一个可执行文件);但它不会生成新的进程
exec函数族的函数执行成功不返回,调用失败会返回-1 , 从调用点接着往下执行;
execl函数
/*#include <unistd.h>int execl(const char *path, const char *arg, ...);参数:path - 需要指定的可执行文件路径/名称a.out /home/nowcoder/a.out(推荐)arg - 可执行文件所需的参数列表1st - 一般没有作用,一般写执行程序名称参数列表必须以NULL结束(哨兵)返回值:出错返回-1 并设置errno
*/
#include <unistd.h>
#include <iostream>
using namespace std;int main(){// 创建一个子进程 在子进程执行exec函数族中的函数pid_t pid = fork();if(pid>0){cout<<"我是你爹"<<" "<<getpid()<<endl;sleep(1);}else if(pid == 0){execl("hello" , "hello" , NULL);cout<<"我是你儿子"<<" "<<getpid()<<endl;}for(int i = 0 ; i<3 ; i++){cout<<i<<" "<<getpid()<<endl;}return 0;
}
execlp 从环境变量查可执行文件
/*#include <unistd.h>int execlp(const char *file, const char *arg,);-- 会到环境变量中查可执行文件 找不到执行失败参数:file - 需要指定的可执行文件的文件名a.out /home/nowcoder/a.out(推荐)arg - 可执行文件所需的参数列表1st - 一般没有作用,一般写执行程序名称参数列表必须以NULL结束(哨兵)返回值:出错返回-1 并设置errno
*/
#include <unistd.h>
#include <iostream>
using namespace std;int main(){// 创建一个子进程 在子进程执行exec函数族中的函数pid_t pid = fork();if(pid>0){cout<<"我是你爹"<<" "<<getpid()<<endl;sleep(1);}else if(pid == 0){execlp("ps" , "ps" , "aux" , NULL);cout<<"我是你儿子"<<" "<<getpid()<<endl;}for(int i = 0 ; i<3 ; i++){cout<<i<<" "<<getpid()<<endl;}return 0;
}
进程退出、孤儿进程、僵尸进程
进程退出:exit(标准C库)、_exit(linux系统函数)
/*#include <stdlib.h>void exit(int status);#include <unistd.h>void _exit(int status);status - 进程退出时的状态信息 父进程回收子进程资源时可以获取
*/
#include <iostream>
#include <stdlib.h>
#include <unistd.h>
using namespace std;
int main(){cout<<"hello"<<endl;cout<<"world";// exit(0); // hello world_exit(0); // helloreturn 0;
}
孤儿进程:父进程运行结束,子进程还在运行 -> 孤儿进程;
- 内核会把孤儿进程的父进程设置为init , init进程会循环wait()退出的子进程;
#include <iostream>
#include <sys/types.h>
#include <unistd.h>
using namespace std;
int main(){// 创建子进程pid_t pid = fork();// 判断父子进程if(pid>0){cout<<pid<<endl;cout<<"父进程 - 进程号:"<<getpid()<<endl;}else if(pid == 0){sleep(1);cout<<"子进程 - 进程号:"<<getpid()<<endl;cout<<"子进程 - 父进程:"<<getppid()<<endl;}for(int i = 0 ; i<3 ; i++){cout<<"i: "<<i<<" "<<getpid()<<endl;}return 0;
}
父进程死亡后切换到前台(出现上述现象);
僵尸进程:进程终止,可以释放用户区的数据,内核区的PCB没办法自己释放,需要父进程进行释放。如果父进程尚未回收,子进程残留资源存放于内核;变成僵尸进程;
不能被kill -9 杀死;
#include <iostream>
#include <sys/types.h>
#include <unistd.h>
using namespace std;
int main(){// 创建子进程pid_t pid = fork();// 判断父子进程if(pid>0){cout<<pid<<endl;while(1){cout<<"父进程 - 进程号:"<<getpid()<<endl;sleep(1);}}else if(pid == 0){cout<<"子进程 - 进程号:"<<getpid()<<endl;cout<<"子进程 - 父进程:"<<getppid()<<endl;}for(int i = 0 ; i<3 ; i++){cout<<"i: "<<i<<" "<<getpid()<<endl;}return 0;
}
处理方法:
1. 父进程调wait()/waitpid()
2. 杀死父进程,让Init接管子进程进行释放处理;
wait/waitpid 函数
wait()函数会阻塞,waitpid()可以设置不阻塞,并且waitpid()可以指定等待哪个子进程结束;
一次wait/waitpid只能清理一个子进程,清理多个子进程应该使用循环;
wait()
调用wait的进程会被挂起,直到其一个子进程退出或遇到不可忽略的信号;
如果其没有子进程或者子进程都结束了会立刻返回-1;
/*#include <sys/types.h>#include <sys/wait.h>pid_t wait(int *wstatus);等待任一子进程结束 然后回收子进程资源;参数:wstatus - 进程退出时的状态信息(传出参数)返回值:成功 - 被回收的子进程id失败 - -11. 所有的子进程都结束2. 调用函数失败
*/
#include <iostream>
#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>using namespace std;int main(){// 创建5个子进程pid_t pid;for(int i = 0 ; i<5 ; i++){pid = fork();if(pid == 0){break;}}if(pid>0){while(1){cout<<"我是你爹: "<<getpid()<<endl;int ret = wait(NULL);// NULL 不获取状态if(ret == -1){break;}cout<<"捕获到了子进程:"<<ret<<endl;sleep(2);}}else if(pid == 0){while(1){cout<<"我是你儿子: "<<getpid()<<endl;sleep(2);}}return 0;
}
waitpid()
/*#include <sys/types.h>#include <sys/wait.h>pid_t waitpid(pid_t pid, int *wstatus, int options);功能:回收指定进程号子进程 设置阻塞/非阻塞参数:pid<-1 - 回收某个进程组的子进程 组id == abs(pid)-1 - 回收所有子进程 相当于wait()0 - 回收当前进程组的所有子进程>0 - 回收指定子进程ID进程watatus - 同waitoptions0 - 阻塞WNOHANG - 非阻塞返回值:>0 - 子进程ID=0 - options=WNOHANG 表示还有子进程-1 - 错误/没有子进程waitpid(-1 , __ , 0) = wait(__);
*/#include <iostream>
#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>
#include <stdlib.h>using namespace std;int main(){// 创建5个子进程pid_t pid;for(int i = 0 ; i<5 ; i++){pid = fork();if(pid == 0){break;}}if(pid>0){while(1){cout<<"我是你爹: "<<getpid()<<endl;sleep(2);int st;// int ret = waitpid(-1 , &st , 0);int ret = waitpid(-1 , &st , WNOHANG);if(ret == -1){break;}if(ret == 0){cout<<"他妈的怎么还有子进程"<<endl;continue;}else{if(WIFEXITED(st)){cout<<"退出的状态码:"<<WEXITSTATUS(st)<<endl;}if(WIFSIGNALED(st)){cout<<"被哪个信号干掉:"<<WTERMSIG(st)<<endl;}cout<<"捕获到了子进程:"<<ret<<endl;}}}else if(pid == 0){while(1){cout<<"我是你儿子: "<<getpid()<<endl;sleep(2);}}return 0;
}
相关文章:

linux并发服务器 —— 多进程并发(四)
进程概述 程序是包含一系列信息的文件,描述了如何在运行时创建一个进程; 进程是正在运行的程序的实例,可以用一个程序来创建多个进程; 用户内存空间包含程序代码以及代码所使用的变量,内核数据结构用于维护进程状态…...

华为云Stack的学习(四)
五、Service OM资源管理 1.Service OM简介 1.1 Service OM介绍 在华为云Stack解决方案中,Service OM是FusionSphere OpenStack的操作管理界面,是资源池(计算、存储、网络)以及基础云服务的管理工具。 1.2 Service OM定位 Serv…...

Midjourney 完整版教程(从账号注册到设计应用)
目录 一、Midjourney 介绍 二、Midjourney 的AI出图示例 三、手把手教你上手Midjourney 1、账号&初始化 1.1 账号注册登录 1.2 账号付费 1.3 账号初始化 2、Midjourney的基础设置 3、Midjourney 出图步骤。 (一)直接描述出图 (二)垫图生图。 4、Midjourney的…...
保护香港服务器的方法
保护香港服务器的方法 当你把一个香港服务器完全留给一个组织、应用程序或个人使用时,它被称为香港服务器租用。在这种类型的主机配置中,客户端将会借出整个服务器,并且不允许其他任何人使用它。 如果您计划使用香港服务器,安全性…...

Redis——》Pipeline
推荐链接: 总结——》【Java】 总结——》【Mysql】 总结——》【Redis】 总结——》【Kafka】 总结——》【Spring】 总结——》【SpringBoot】 总结——》【MyBatis、MyBatis-Plus】 总结——》【Linux】 总结——》【MongoD…...

Git企业开发控制理论和实操-从入门到深入(六)|多人协作开发
前言 那么这里博主先安利一些干货满满的专栏了! 首先是博主的高质量博客的汇总,这个专栏里面的博客,都是博主最最用心写的一部分,干货满满,希望对大家有帮助。 高质量博客汇总 然后就是博主最近最花时间的一个专栏…...
通过 ChatGPT 学习 Python
先决条件 您需要一个 OpenAI 帐户才能开始与 ChatGPT 交互。如果您还没有这样做,请在 OpenAI 网站上注册一个帐户。 什么是 ChatGPT? GPT(Generative Pre-training Transformer)是 OpenAI 开发的一种语言模型,它使用深度学习技术生成类似人类的文本。ChatGPT 是 GPT 模…...

开发卡牌gamefi游戏需要多少钱?
卡牌游戏作为一种受欢迎的游戏形式,吸引了众多开发者的关注。然而,开发一款成功的卡牌游戏需要全面考虑多个方面的因素,其中之一就是资金投入。本文将从专业性和投入回报的角度,探讨开发一款卡牌游戏所需的资金投入。 一、专业性的…...

linux服务TCP参数配置
Linux TCP参数配置 阿里云规范 1.【推荐】高并发服务器建议调小 TCP 协议的 time_wait 超时时间。 说明:操作系统默认 240 秒后,才会关闭处于 time_wait 状态的连接,在高并发访问下,服务器端会因为处于 time_wait 的连接数太多&am…...

部署Spring Boot项目
上传jar包 之前在新建Spring Boot项目[1]使用mvn install的方式,已经构建出jar包。 通过scp或rz/sz,将该jar包上传到服务器 执行java -jar hello-0.0.1-SNAPSHOT.jar,发生如下报错: Exception in thread "main" java.lang.Unsuppo…...

Java 中数据结构LinkedList的用法
LinkList 链表(Linked list)是一种常见的基础数据结构,是一种线性表,但是并不会按线性的顺序存储数据,而是在每一个节点里存到下一个节点的地址。 链表可分为单向链表和双向链表。 一个单向链表包含两个值: 当前节点…...

jmeter递增压测线程组配置
jmeter递增压测线程组配置 新建线程组线程组参数详解及填写其他指标设置 新建线程组 操作位置如图: 线程组参数详解及填写 其他指标设置 其他指标设置可参考另一篇文章: 链接: jmeter 在linux服务器中执行性能测试、监听服务器资源指标...
hutool工具
Hutool是一个Java工具包 参考:https://www.hutool.cn/ <dependency><groupId>cn.hutool</groupId><artifactId>hutool-all</artifactId><version>4.6.3</version> </dependency>Convert类型转换工具类 //转换为…...

Redis.conf 配置文件详解
1、units 单位 配置大小单位,开头定义了一些基本的度量单位,只支持 bytes,不支持bit,并且对大小写 不敏感。 2、INCLUDES 包含 类似于 Spring 配置文件,可以通过 includes 包含,redis.conf 可以作为总文件…...
linux磁盘空间满了
登录服务器,通过df -Hl查看 确定告警信息一致,接着是找到占用空间大目录或文件 一种比较笨的方法是,在根目录下,通过du -sh命令,列出各目录所占空间大小 之后再用同样的方法继续到对应目录下去找 再相对高效一点的…...

Ansible项目实战管理/了解项目环境/项目管理
一,项目环境 1.项目基础 项目过程 调研阶段 设计阶段 开发阶段 测试阶段 运营阶段 2.项目环境 个人开发环境 公司开发环境 项目测试环境 项目预发布环境 灰度环境:本身是生产环境,安装项目规划,最终所有的生产环境都发…...

hadoop 学习:mapreduce 入门案例一:WordCount 统计一个文本中单词的个数
一 需求 这个案例的需求很简单 现在这里有一个文本wordcount.txt,内容如下 现要求你使用 mapreduce 框架统计每个单词的出现个数 这样一个案例虽然简单但可以让新学习大数据的同学熟悉 mapreduce 框架 二 准备工作 (1)创建一个 maven 工…...
vue2项目中el-input单独使用max和maxlength不生效问题
vue2项目中el-input单独使用max和maxlength不生效问题 今天在vue2的项目中使用element中的<el-input>组件,因为没有使用form所以max和maxlength属性没有生效,下面是解决办法 <el-input placeholder"请输入" v-model"holeDat…...

源码角度看待线程池的执行流程
文章目录 前言一、线程池的相关接口和实现类1.Executor接口2.ExecutorService接口3.AbstractExecutorService接口4.ThreadPoolExecutor 实现类 二、ThreadPoolExecutor源码解析1.Worker内部类2.execute()方法3.addWorker()方法 总结 前言 线程池内部维护了若干个线程ÿ…...

我们的第一个 Qt 窗口程序
Qt 入门实战教程(目录) Windows Qt 5.12.10下载与安装 为何使用Qt Creator开发QT 本文介绍用Qt自带的集成开发工具Qt Creator创建Qt默认的窗口程序。 本文不需要你另外安装Visual Studio 2022这样的集成开发环境,也不需要你再在Visual St…...

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器的上位机配置操作说明
LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器专为工业环境精心打造,完美适配AGV和无人叉车。同时,集成以太网与语音合成技术,为各类高级系统(如MES、调度系统、库位管理、立库等)提供高效便捷的语音交互体验。 L…...
进程地址空间(比特课总结)
一、进程地址空间 1. 环境变量 1 )⽤户级环境变量与系统级环境变量 全局属性:环境变量具有全局属性,会被⼦进程继承。例如当bash启动⼦进程时,环 境变量会⾃动传递给⼦进程。 本地变量限制:本地变量只在当前进程(ba…...

突破不可导策略的训练难题:零阶优化与强化学习的深度嵌合
强化学习(Reinforcement Learning, RL)是工业领域智能控制的重要方法。它的基本原理是将最优控制问题建模为马尔可夫决策过程,然后使用强化学习的Actor-Critic机制(中文译作“知行互动”机制),逐步迭代求解…...
Qwen3-Embedding-0.6B深度解析:多语言语义检索的轻量级利器
第一章 引言:语义表示的新时代挑战与Qwen3的破局之路 1.1 文本嵌入的核心价值与技术演进 在人工智能领域,文本嵌入技术如同连接自然语言与机器理解的“神经突触”——它将人类语言转化为计算机可计算的语义向量,支撑着搜索引擎、推荐系统、…...
是否存在路径(FIFOBB算法)
题目描述 一个具有 n 个顶点e条边的无向图,该图顶点的编号依次为0到n-1且不存在顶点与自身相连的边。请使用FIFOBB算法编写程序,确定是否存在从顶点 source到顶点 destination的路径。 输入 第一行两个整数,分别表示n 和 e 的值(1…...

Maven 概述、安装、配置、仓库、私服详解
目录 1、Maven 概述 1.1 Maven 的定义 1.2 Maven 解决的问题 1.3 Maven 的核心特性与优势 2、Maven 安装 2.1 下载 Maven 2.2 安装配置 Maven 2.3 测试安装 2.4 修改 Maven 本地仓库的默认路径 3、Maven 配置 3.1 配置本地仓库 3.2 配置 JDK 3.3 IDEA 配置本地 Ma…...

处理vxe-table 表尾数据是单独一个接口,表格tableData数据更新后,需要点击两下,表尾才是正确的
修改bug思路: 分别把 tabledata 和 表尾相关数据 console.log() 发现 更新数据先后顺序不对 settimeout延迟查询表格接口 ——测试可行 升级↑:async await 等接口返回后再开始下一个接口查询 ________________________________________________________…...

Golang——6、指针和结构体
指针和结构体 1、指针1.1、指针地址和指针类型1.2、指针取值1.3、new和make 2、结构体2.1、type关键字的使用2.2、结构体的定义和初始化2.3、结构体方法和接收者2.4、给任意类型添加方法2.5、结构体的匿名字段2.6、嵌套结构体2.7、嵌套匿名结构体2.8、结构体的继承 3、结构体与…...
LRU 缓存机制详解与实现(Java版) + 力扣解决
📌 LRU 缓存机制详解与实现(Java版) 一、📖 问题背景 在日常开发中,我们经常会使用 缓存(Cache) 来提升性能。但由于内存有限,缓存不可能无限增长,于是需要策略决定&am…...
提升移动端网页调试效率:WebDebugX 与常见工具组合实践
在日常移动端开发中,网页调试始终是一个高频但又极具挑战的环节。尤其在面对 iOS 与 Android 的混合技术栈、各种设备差异化行为时,开发者迫切需要一套高效、可靠且跨平台的调试方案。过去,我们或多或少使用过 Chrome DevTools、Remote Debug…...