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.mzph.cn/news/59413.shtml

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

相关文章

超声波俱乐部分享:AI冷静期,创业者们应该做什么?

8月26日&#xff0c;2023年第十一期超声波俱乐部内部分享会在北京望京举行。本期的主题是&#xff1a;AI冷静期&#xff0c;创业者们应该做什么&#xff1f; 到场的嘉宾有&#xff1a; 超声波创始人杨子超&#xff0c;超声波联合创始人、和牛商业创始人刘思雨&#xff0c;中国…

探讨uniapp的页面问题

1 新建页面 uni-app中的页面&#xff0c;默认保存在工程根目录下的pages目录下。 每次新建页面&#xff0c;均需在pages.json中配置pages列表&#xff1b; 未在pages.json -> pages 中注册的页面&#xff0c;uni-app会在编译阶段进行忽略。pages.json的完整配置参考&am…

知更鸟语音训练

现在哪还有人读小说、看视频&#xff0c;谁还用真人朗读呢&#xff1f; 现在给大家介绍&#xff0c;假人朗读是怎么来的&#xff0c;提供一些音频&#xff0c;进行训练&#xff0c;然后就能合成属于自己的音频了。这里只讲训练部分&#xff0c;使用请自己看github知更鸟&#…

Python2021年06月Python二级 -- 编程题解析

题目一 没有重复数字的两位数统计 编写一段程序&#xff0c;实现下面的功能: (1)检查所有的两位数; (2)程序自动分析两位数上的个位与十位上的数字是否相同&#xff0c;相同则剔除&#xff0c; 不同则保留(例:12符合本要求&#xff0c;个位是2&#xff0c;十位是1&#xff0c;两…

what(): NCCL Error 1: unhandled cuda error解决方法

文章目录 遇到问题解决方法参考 遇到问题 运行项目&#xff1a;ACL2021的一篇工作&#xff0c;LM-BFF (Better Few-shot Fine-tuning of Language Models) https://github.com/princeton-nlp/LM-BFF 遇到环境问题。 我的机器环境如下&#xff1a; 服务器上CUDA版本为11.4 GPU…

常见的下载方式

一. 使用 window.open() 使用场景 // 1. 先封装一个实习下载的函数 export const download (path) > {window.open(下载的接口&#xff0c;例如&#xff1a;/fs/download?path path) } // 2. 使用&#xff1a;在需要下载的地方调用download函数&#xff0c;传入下载的u…

视频批量剪辑矩阵分发系统源码开源分享----基于PHP语言

批量剪辑视频矩阵分发&#xff1a; 短视频seo主要基于抖音短视频平台&#xff0c;为企业实现多账号管理&#xff0c;视频分发&#xff0c;视频批量剪辑&#xff0c;抖音小程序搭建&#xff0c;企业私域转化等&#xff0c;本文主要介绍短视频矩阵系统抖音小程序开发详细及注意事…

linux创建进程

linux创建进程 准备工作 准备工作 在Ubuntu64系统上 1、安装GCC和Make工具 编译器GCC&#xff1a;把C源码转为二进制程序 Make&#xff1a;自动编译多源文件项目 sudo apt-get update #更新存储库 sudo apt-get install build-essential #安装build-essential包 gcc --versio…

docker之 Consul(注册与发现)

目录 一、什么是服务注册与发现&#xff1f; 二、什么是consul 三、consul 部署 3.1建立Consul服务 3.1.1查看集群状态 3.1.2通过 http api 获取集群信息 3.2registrator服务器 3.2.1安装 Gliderlabs/Registrator 3.2.2测试服务发现功能是否正常 3.2.3验证 http 和 ng…

关于异数OS服务器CPU效能分析工具

该工具发布背景 近年来&#xff0c;国产服务器CPU产业的逐渐发展&#xff0c;但由于专业性较差&#xff0c;与国外存在40年以上技术差距&#xff0c;一些服务器CPU厂商利用信息差来制造一些非专业的数据夸大并虚假宣传混淆视听&#xff0c;成功达到劣币驱良币的目标&#xff0…

【论文笔记】最近看的时空数据挖掘综述整理8.27

Deep Learning for Spatio-Temporal Data Mining: A Survey 被引用次数&#xff1a;392 [Submitted on 11 Jun 2019 (v1), last revised 24 Jun 2019 (this version, v2)] 主要内容&#xff1a; 该论文是一篇关于深度学习在时空数据挖掘中的应用的综述。论文首先介绍了时空数…

react css 污染解决方法

上代码 .m-nav-bar {background: #171a21;.content {height: 104px;margin: 0px auto;} }import React from "react"; import styles from ./css.module.scssexport default class NavBar extends React.Component<any, any> {constructor (props: any) {supe…

数组(个人学习笔记黑马学习)

一维数组 1、定义方式 #include <iostream> using namespace std;int main() {//三种定义方式//1.int arr[5];arr[0] 10;arr[1] 20;arr[2] 30;arr[3] 40;arr[4] 50;//访问数据元素/*cout << arr[0] << endl;cout << arr[1] << endl;cout &l…

Redis怎么测试?

有些测试朋友来问我&#xff0c;redis要怎么测试&#xff1f;首先我们需要知道&#xff0c;redis是什么&#xff1f;它能做什么&#xff1f; redis是一个key-value类型的高速存储数据库。 redis常被用做&#xff1a;缓存、队列、发布订阅等。 所以&#xff0c;“redis要怎么…

web自动化框架:selenium学习使用操作大全(Python版)

目录 一、浏览器驱动下载二、selenium-python安装&#xff08;打开网站、操作元素&#xff09;三、网页解析&#xff08;HTML、xpath&#xff09;四、selenium基本操作1、元素定位八种方法2、元素动态定位3、iframe切换4、填充表单_填充文本框5、填充表单_单选按钮6、填充表单_…

山海优选卷轴模式源码实例分享

山海优选通过与绿色有机产品供应商和生产商的合作&#xff0c;建立起稳定的供应链关系&#xff0c;确保产品的绿色、有机、健康。通过产品认证和检测&#xff0c;为会员提供可信赖的绿色产品。积分激励机制设计 山海优选引入“绿色积分”机制&#xff0c;通过会员购买产品、参与…

docker 搭建私有仓库和制作镜像

目录 1、使用mysql:5.6和 owncloud 镜像&#xff0c;构建一个个人网盘。 1.1 启动mysql镜像 1.2 启动owncloud镜像 1.3 浏览器访问 1.4 总结 2、安装搭建私有仓库 Harbor 2.1 下载docker-compose并赋予执行权限 2.2 磁盘挂载&#xff0c;保存harbor 2.3 修改配置文件…

4G WiFi LoRa无线外夹式超声波管道流量计MQTT/http协议 json数据说明

ip&#xff1a;114.128.112.131 port&#xff1a;1883 uname&#xff1a;scwl_flowmeter pwd&#xff1a;b123 topic&#xff1a;iot/data/scwlflowmeter { “deviceId”:“设备序列号”, “flow”:“瞬时流量&#xff08;浮点数&#xff09;”, “heatFlow”:“瞬时热流量&am…

Web 自动化神器 TestCafe—页面高级操作篇

♥ 前 言 在【Web 自动化神器 TestCafe — 页面基本操作篇】这篇文章中我们介绍了TestCafe页面交互的一些基本使用 Web 自动化神器 TestCafe — 页面基本操作篇 这篇文章接着上一篇来给大家介绍一下 TestCafe 页面交互的一些高级操作。 一、鼠标拖拽 鼠标拖拽鼠标拖拽 …