当前位置: 首页 > news >正文

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并发服务器 —— 多进程并发(四)

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

华为云Stack的学习(四)

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

Midjourney 完整版教程(从账号注册到设计应用)

目录 一、Midjourney 介绍 二、Midjourney 的AI出图示例 三、手把手教你上手Midjourney 1、账号&初始化 1.1 账号注册登录 1.2 账号付费 1.3 账号初始化 2、Midjourney的基础设置 3、Midjourney 出图步骤。 (一)直接描述出图 (二)垫图生图。 4、Midjourney的…...

保护香港服务器的方法

保护香港服务器的方法 当你把一个香港服务器完全留给一个组织、应用程序或个人使用时&#xff0c;它被称为香港服务器租用。在这种类型的主机配置中&#xff0c;客户端将会借出整个服务器&#xff0c;并且不允许其他任何人使用它。 如果您计划使用香港服务器&#xff0c;安全性…...

Redis——》Pipeline

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

Git企业开发控制理论和实操-从入门到深入(六)|多人协作开发

前言 那么这里博主先安利一些干货满满的专栏了&#xff01; 首先是博主的高质量博客的汇总&#xff0c;这个专栏里面的博客&#xff0c;都是博主最最用心写的一部分&#xff0c;干货满满&#xff0c;希望对大家有帮助。 高质量博客汇总 然后就是博主最近最花时间的一个专栏…...

通过 ChatGPT 学习 Python

先决条件 您需要一个 OpenAI 帐户才能开始与 ChatGPT 交互。如果您还没有这样做,请在 OpenAI 网站上注册一个帐户。 什么是 ChatGPT? GPT(Generative Pre-training Transformer)是 OpenAI 开发的一种语言模型,它使用深度学习技术生成类似人类的文本。ChatGPT 是 GPT 模…...

开发卡牌gamefi游戏需要多少钱?

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

linux服务TCP参数配置

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

部署Spring Boot项目

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

Java 中数据结构LinkedList的用法

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

jmeter递增压测线程组配置

jmeter递增压测线程组配置 新建线程组线程组参数详解及填写其他指标设置 新建线程组 操作位置如图&#xff1a; 线程组参数详解及填写 其他指标设置 其他指标设置可参考另一篇文章&#xff1a; 链接: jmeter 在linux服务器中执行性能测试、监听服务器资源指标...

hutool工具

Hutool是一个Java工具包 参考&#xff1a;https://www.hutool.cn/ <dependency><groupId>cn.hutool</groupId><artifactId>hutool-all</artifactId><version>4.6.3</version> </dependency>Convert类型转换工具类 //转换为…...

Redis.conf 配置文件详解

1、units 单位 配置大小单位&#xff0c;开头定义了一些基本的度量单位&#xff0c;只支持 bytes&#xff0c;不支持bit&#xff0c;并且对大小写 不敏感。 2、INCLUDES 包含 类似于 Spring 配置文件&#xff0c;可以通过 includes 包含&#xff0c;redis.conf 可以作为总文件…...

linux磁盘空间满了

登录服务器&#xff0c;通过df -Hl查看 确定告警信息一致&#xff0c;接着是找到占用空间大目录或文件 一种比较笨的方法是&#xff0c;在根目录下&#xff0c;通过du -sh命令&#xff0c;列出各目录所占空间大小 之后再用同样的方法继续到对应目录下去找 再相对高效一点的…...

Ansible项目实战管理/了解项目环境/项目管理

一&#xff0c;项目环境 1.项目基础 项目过程 调研阶段 设计阶段 开发阶段 测试阶段 运营阶段 2.项目环境 个人开发环境 公司开发环境 项目测试环境 项目预发布环境 灰度环境&#xff1a;本身是生产环境&#xff0c;安装项目规划&#xff0c;最终所有的生产环境都发…...

hadoop 学习:mapreduce 入门案例一:WordCount 统计一个文本中单词的个数

一 需求 这个案例的需求很简单 现在这里有一个文本wordcount.txt&#xff0c;内容如下 现要求你使用 mapreduce 框架统计每个单词的出现个数 这样一个案例虽然简单但可以让新学习大数据的同学熟悉 mapreduce 框架 二 准备工作 &#xff08;1&#xff09;创建一个 maven 工…...

vue2项目中el-input单独使用max和maxlength不生效问题

vue2项目中el-input单独使用max和maxlength不生效问题 今天在vue2的项目中使用element中的<el-input>组件&#xff0c;因为没有使用form所以max和maxlength属性没有生效&#xff0c;下面是解决办法 <el-input placeholder"请输入" v-model"holeDat…...

源码角度看待线程池的执行流程

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

我们的第一个 Qt 窗口程序

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

利用最小二乘法找圆心和半径

#include <iostream> #include <vector> #include <cmath> #include <Eigen/Dense> // 需安装Eigen库用于矩阵运算 // 定义点结构 struct Point { double x, y; Point(double x_, double y_) : x(x_), y(y_) {} }; // 最小二乘法求圆心和半径 …...

AI-调查研究-01-正念冥想有用吗?对健康的影响及科学指南

点一下关注吧&#xff01;&#xff01;&#xff01;非常感谢&#xff01;&#xff01;持续更新&#xff01;&#xff01;&#xff01; &#x1f680; AI篇持续更新中&#xff01;&#xff08;长期更新&#xff09; 目前2025年06月05日更新到&#xff1a; AI炼丹日志-28 - Aud…...

线程与协程

1. 线程与协程 1.1. “函数调用级别”的切换、上下文切换 1. 函数调用级别的切换 “函数调用级别的切换”是指&#xff1a;像函数调用/返回一样轻量地完成任务切换。 举例说明&#xff1a; 当你在程序中写一个函数调用&#xff1a; funcA() 然后 funcA 执行完后返回&…...

五年级数学知识边界总结思考-下册

目录 一、背景二、过程1.观察物体小学五年级下册“观察物体”知识点详解&#xff1a;由来、作用与意义**一、知识点核心内容****二、知识点的由来&#xff1a;从生活实践到数学抽象****三、知识的作用&#xff1a;解决实际问题的工具****四、学习的意义&#xff1a;培养核心素养…...

Robots.txt 文件

什么是robots.txt&#xff1f; robots.txt 是一个位于网站根目录下的文本文件&#xff08;如&#xff1a;https://example.com/robots.txt&#xff09;&#xff0c;它用于指导网络爬虫&#xff08;如搜索引擎的蜘蛛程序&#xff09;如何抓取该网站的内容。这个文件遵循 Robots…...

css的定位(position)详解:相对定位 绝对定位 固定定位

在 CSS 中&#xff0c;元素的定位通过 position 属性控制&#xff0c;共有 5 种定位模式&#xff1a;static&#xff08;静态定位&#xff09;、relative&#xff08;相对定位&#xff09;、absolute&#xff08;绝对定位&#xff09;、fixed&#xff08;固定定位&#xff09;和…...

基于Java+VUE+MariaDB实现(Web)仿小米商城

仿小米商城 环境安装 nodejs maven JDK11 运行 mvn clean install -DskipTestscd adminmvn spring-boot:runcd ../webmvn spring-boot:runcd ../xiaomi-store-admin-vuenpm installnpm run servecd ../xiaomi-store-vuenpm installnpm run serve 注意&#xff1a;运行前…...

【p2p、分布式,区块链笔记 MESH】Bluetooth蓝牙通信 BLE Mesh协议的拓扑结构 定向转发机制

目录 节点的功能承载层&#xff08;GATT/Adv&#xff09;局限性&#xff1a; 拓扑关系定向转发机制定向转发意义 CG 节点的功能 节点的功能由节点支持的特性和功能决定。所有节点都能够发送和接收网格消息。节点还可以选择支持一个或多个附加功能&#xff0c;如 Configuration …...

【Linux手册】探秘系统世界:从用户交互到硬件底层的全链路工作之旅

目录 前言 操作系统与驱动程序 是什么&#xff0c;为什么 怎么做 system call 用户操作接口 总结 前言 日常生活中&#xff0c;我们在使用电子设备时&#xff0c;我们所输入执行的每一条指令最终大多都会作用到硬件上&#xff0c;比如下载一款软件最终会下载到硬盘上&am…...

【堆垛策略】设计方法

堆垛策略的设计是积木堆叠系统的核心&#xff0c;直接影响堆叠的稳定性、效率和容错能力。以下是分层次的堆垛策略设计方法&#xff0c;涵盖基础规则、优化算法和容错机制&#xff1a; 1. 基础堆垛规则 (1) 物理稳定性优先 重心原则&#xff1a; 大尺寸/重量积木在下&#xf…...