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;
}

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

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

相关文章

Spring Cloud--从零开始搭建微服务基础环境【三】

&#x1f600;前言 本篇博文是关于Spring Cloud–从零开始搭建微服务基础环境【三】&#xff0c;希望你能够喜欢 &#x1f3e0;个人主页&#xff1a;晨犀主页 &#x1f9d1;个人简介&#xff1a;大家好&#xff0c;我是晨犀&#xff0c;希望我的文章可以帮助到大家&#xff0c;…

深度学习推荐系统(五)DeepCrossing模型及其在Criteo数据集上的应用

深度学习推荐系统(五)Deep&Crossing模型及其在Criteo数据集上的应用 在2016年&#xff0c; 随着微软的Deep Crossing&#xff0c; 谷歌的Wide&Deep以及FNN、PNN等一大批优秀的深度学习模型被提出&#xff0c; 推荐系统全面进入了深度学习时代&#xff0c; 时至今日&am…

Qt应用开发(基础篇)——对话框窗口 QDialog

一、前言 QDialog类继承于QWidget&#xff0c;是Qt基于对话框窗口(消息窗口QMessageBox、颜色选择窗口QColorDialog、文件选择窗口QFileDialog等)的基类。 QDialog窗口是顶级的窗口&#xff0c;一般情况下&#xff0c;用来当做用户短期任务(确认、输入、选择)或者和用户交流(提…

深度学习怎么学?

推荐这本小白看的《深度学习&#xff1a;从基础到实践&#xff08;上下册&#xff09;》。 深度学习&#xff1a;从基础到实践&#xff08;上下册&#xff09; 深入浅出的讲述了深度学习的基本概念与理论知识&#xff0c;不涉及复杂的数学内容&#xff0c;零基础小白也能轻松掌…

QT day1登录界面设计

要设计如下图片&#xff1a; 代码如下&#xff1a; main.cpp widget.h widget.cpp 运行效果&#xff1a; 2&#xff0c;思维导图

任务执行和调度----Spring线程池/Quartz

定时任务 在服务器中可能会有定时任务&#xff0c;但是不知道分布式系统下次会访问哪一个服务器&#xff0c;所以服务器中的任务就是相同的&#xff0c;这样会导致浪费。使用Quartz可以解决这个问题。 JDK线程池 RunWith(SpringRunner.class) SpringBootTest ContextConfi…

Spark-Core核心算子

文章目录 一、数据源获取1、从集合中获取2、从外部存储系统创建3、从其它RDD中创建4、分区规则—load数据时 二、转换算子(Transformation)1、Value类型1.1 map()_1.2 mapPartitions()1.3 mapPartitionsWithIndex(不常用)1.4 filterMap()_扁平化&#xff08;合并流&#xff09;…

卡特兰数和算法

在组合数学中&#xff0c;卡特兰数是一系列自然数&#xff0c;出现在各种组合计数问题中&#xff0c;通常涉及递归定义的对象。它们以比利时数学家尤金查尔斯卡特兰&#xff08;Eugne Charles Catalan&#xff09;的名字命名。 卡特兰数序列是1, 1, 2, 5, 14, 42......&#xf…

java.sql.SQLException: com.mysql.cj.jdbc.Driver

这篇文章分享一下Springboot整合Elasticsearch时遇到的一个问题&#xff0c;项目正常启动&#xff0c;但是查询数据库的时候发生了一个异常java.sql.SQLException: com.mysql.cj.jdbc.Driver java.sql.SQLException: com.mysql.cj.jdbc.Driverat com.alibaba.druid.util.JdbcU…

【德哥说库系列】-ASM管理Oracle 19C单实例部署

&#x1f4e2;&#x1f4e2;&#x1f4e2;&#x1f4e3;&#x1f4e3;&#x1f4e3; 哈喽&#xff01;大家好&#xff0c;我是【IT邦德】&#xff0c;江湖人称jeames007&#xff0c;10余年DBA及大数据工作经验 一位上进心十足的【大数据领域博主】&#xff01;&#x1f61c;&am…

微软 Turing Bletchley v3视觉语言模型更新:必应搜索图片更精准

据微软新闻稿透露&#xff0c;在推出第三代Turing Bletchley视觉语言模型后&#xff0c;微软计划逐步将其整合到Bing等相关产品中&#xff0c;以提供更出色的图像搜索体验。这款模型最初于2021年11月面世&#xff0c;并在2022年秋季开始邀请用户测试。 凭借用户的反馈和建议&am…

mapboxGL3新特性介绍

概述 8月7日&#xff0c;mapboxGL发布了3版本的更新&#xff0c;本文带大家一起来看看mapboxGL3有哪些新的特性。 新特新 如上图所示&#xff0c;是mapboxGL官网关于新版的介绍&#xff0c;大致翻译如下&#xff1a; 增强了web渲染的质量、便捷程度以及开发人员体验&#xff…

【云计算•云原生】5.云原生之初识DevOps

文章目录 1.DevOps背景2.DevOps概念3.DevOps工具链 1.DevOps背景 软件开发必须包含两个团队&#xff1a;开发团队和运维团队 开发团队负责开发项目&#xff0c;系统迭代更新运维团队负责项目测试以及部署上线&#xff0c;维持系统稳定运行 一个软件周期中是由这两个团队相互…

buildroot修改内核防止清理重新加载办法

当你使用 Buildroot 构建 Linux 内核时&#xff0c;如果对内核文件进行了手动修改&#xff0c;重新执行 Buildroot 的构建过程将会覆盖你所做的修改。这是因为 Buildroot会根据配置重新下载、提取和编译内核。 为了避免在重新构建时覆盖你的修改&#xff0c;可以采取以下两种方…

数据可视化与数字孪生:理解两者的区别

在数字化时代&#xff0c;数据技术正在引领创新&#xff0c;其中数据可视化和数字孪生是两个备受关注的概念。尽管它们都涉及数据的应用&#xff0c;但在本质和应用方面存在显著区别。本文带大探讨数据可视化与数字孪生的差异。 概念 数据可视化&#xff1a; 数据可视化是将复…

Windows下将nginx等可执行文件添加为服务

Windows下将nginx等可执行文件添加为服务 为什么将可执行文件添加为服务&#xff1f;将可执行文件添加为服务的步骤步骤 1&#xff1a;下载和安装 Nginx步骤 2&#xff1a;添加为服务方法一&#xff1a;使用 Windows 自带的 sc 命令方法二&#xff1a;使用 NSSM&#xff08;Non…

Vue实战【调整Vue-element-admin中的菜单栏,并添加顶部模块菜单栏】

目录 &#x1f31f;前言&#x1f31f;小伙伴们先看&#x1f31f;实现思路&#x1f31f;具体代码&#x1f31f;最后 &#x1f31f;前言 因为最近在整合公司的项目&#xff0c;需要把所有系统里的功能集成到一个项目里&#xff0c;这样就导致菜单栏目录会特别的多&#xff0c;不…

【MySQL学习笔记】(七)内置函数

内置函数 日期函数示例案例-1案例-2 字符串函数示例 数学函数其他函数 日期函数 示例 获得当前年月日 mysql> select current_date(); ---------------- | current_date() | ---------------- | 2023-09-03 | ---------------- 1 row in set (0.00 sec)获得当前时分秒…

ARM 汇编基础知识

1.为什么学习汇编&#xff1f; 我们在进行嵌入式 Linux 开发的时候是绝对要掌握基本的 ARM 汇编&#xff0c;因为 Cortex-A 芯片一 上电 SP 指针还没初始化&#xff0c; C 环境还没准备好&#xff0c;所以肯定不能运行 C 代码&#xff0c;必须先用汇编语言设置好 C 环境…

HarmonyOS应用开发者高级认证练习题

系列文章目录 HarmonyOS应用开发者基础认证练习题 HarmonyOS应用开发者高级认证练习题 文章目录 系列文章目录前言一、判断二、单选三、多选 前言 本文所有内容来源于个人进行HarmonyOS应用开发者系列认证的学习过程中所做过的练习题&#xff0c;所有答案均是个人作答&#x…