学习系统编程No.18【进程间通信之管道实战】

news2025/1/14 1:22:34

引言:

北京时间:2023/4/11/21:17,今天的文章更新啦!但是还是没有上热榜,所以我们需要继续更文啦!我相信下一篇博客肯定是可以上热榜的,加油!并且今天晚上因为有一节体育课,所以导致现在才开始码字,体育课上教我们打羽毛球,虽然我自从高考到现在已经快一年没打了,但是还是有点实力,但是实力有待恢复,运动其实是非常快乐的,哎!可惜,时间需要用来码字,连打球的时间都挤不出来了,所以让我们趁热打铁,来学习一下有关进程间通信之管道的知识吧!

在这里插入图片描述

深入进程间通信

为什么需要有进程间通信,从以前有关进程的知识,我们可以发现,我们以前学的都是单独一个进程有关的知识,无论是进程终止、进程创建,还是进程等待等知识,所以单独一个进程就像是一个个体,并且明白,个体和个体之间肯定是有一定的联系,所以进程和进程之间也应该存在一定的联系,这个联系的目的,就好比个体和个体之间,是为了促进某些方面的发展,所以进程和进程之间的联系,目的也是为了促进某些方面,例如,节省资源、提高效率等!所以进程间通信的目的就是为单独一个一个的进程之间的联系提供环境,使进程之间可以执行我们想要的一些操作,如:数据传输(一个进程将对应的数据传送给另一个进程)、资源共享(一个进程将资源共享给多个进程使用)、通知事件(一个进程通过系统调用接口参数,来告诉另一个进程有关的事件,如,子进程终止时,通过输出型参数,通知父进程)、进程控制(一个进程为了实时了解另一个进程的状态,此时就需要对该进程进行控制),但是注意: 进程间通信提供的联系环境,这个联系不仅仅涉及进程之间的,本质上都是操作系统来完成,具体如下述所说:

因为进程具有独立性,所以如果想要让两个进程之间能够看到同一份资源,此时要明白,进程和进程之间肯定完成不了这个操作,就需要操作系统来起中间作用,从而让一个进程和另一个进程可以看到同一份资源,最后实现进程间通信的基础条件,两个进程共享同一份资源(文件资源),所以进程间通信的本质不是让进程和进程之间通信,本质只是搭建一个进程和进程之间通信的环境而已,如下图所示:
在这里插入图片描述

从通信理解标准,再依据标准搞定通信
理解了上述有关进程间通信的知识之后,此时我们先摒弃进程,具体来谈谈通信二字,什么是通信,通信二字应该是上世纪非常重要的一个概念,它的提出在当时可能是跨时代的,简简单单从现在就能看出,人类已经离不开通信了,并且在这条道路上已经走得越来越远,从5G通信就可以看出,当今通信技术已经非常的成熟,通信方面的效率性、稳定性、安全性在5G通信上达到新的高度;为什么要有通信这种问题,我们就不谈了,我们重点谈谈为什么通信能够发展的如此迅速,从2G到3G到4G到如今的5G,日新月异,高速发展的背后是因为什么呢?此时就不得不引出跟通信紧紧挂钩的两个字,标准,什么是标准,想必大家都知道,不就是标准吗?哈哈哈!所以什么是标准,你脑袋直接混乱,标准不就是标准吗?所以按照日常生过来理解, 标准就是一个规定,一个规定,大家都必须遵循的准则和依据, 所以明白了这点,我们就知道了,通信能够发展如此迅速的原因,就是因为标准,大家按照标准执行,严格遵守标准,所以明白一个道理,任何事情想要成功,成气候,首先一定要有一个标准,所以通信能够如此成功的原因,就是因为它有一套属于自己的标准:通信标准,感兴趣的同学参考该链接:通信标准详解

了解了上述知识之后,此时我们就明白,任何事情都是存在一个标准的,无论是国家,还是社会,还是世界,只要有人存在,就必然有标准,这就是人类能够不断进步的第一原则,通信不例外,进程间通信更不例外,并且知道,进程间通信的标准主要是两种,一种是System进程间通信标准(本地通信), 一种是POSIX进程间通信标准(网络通信 ,感兴趣的同学参考该链接:进程间通信标准详解,并且明白,进程间通信的种类有多种,在不同的场景或者是环境之下,就有可能使用不同种类的进程间通信,但无论是那个种类的进程间通信,由于有进程间通信标准,所以当在不同的环境或者不同的场景下使用不同种类的进程间通信,进程间通信的实现都大致相同,这就是标准的意义好作用

如何理解管道

明白了上述知识之后,我们就来看看进程间通信最简单的一种,进程间通信之管道的知识

首先要明白,在Linux系统下,| 表示的就是 管道 ,并且当我们使用了管道之后,管道左右对应的指令(函数接口),最终都会变成进程,所以得出结论,你使用一个管道,就会同时生成两个进程,你使用两个管道,就会同时生成三个进程,类推,如下图所示:
在这里插入图片描述
在这里插入图片描述
抽象一个管道,两个进程之间的数据传输,如下图:
在这里插入图片描述
所以如上图可以看出,操作系统在进程间通信的时候,会帮我们创建一个管道,按照Linux系统一切皆文件来看,此时管道本质也就是一个文件,等同于操作系统在进行进程间通信的时候,会帮我们维护一个文件,并且凭借这个文件来实现让两个不同的进程共享一份"资源"的效果 ,这样操作系统就完成了进程间通信环境的搭建,从而让进程之间可以进行通信,那么具体如何通信呢?

有了之前和上述知识,理解如何通信这个问题就非常的简单,没什么多余的知识点和要求,本质上就是让一个进程(who进程)以写入的方式去打开管道文件,将数据拷贝到管道文件中,然后让另一个进程(wc -l)以读的方式打开管道文件,最终从管道文件中读取到数据,就这么简单多了的没有,唯一要注意的是,理解进程的默认输出和默认输入,如下:

注意进程默认打印数据,是向标准输出打印,进程读取数据,默认是从标准输入读取所以此时如果想要实现,让一个进程(who)将数据拷贝到管道文件,而不是标注输出文件,此时就需要将该进程(who)的标准输出重定向到管道文件当中,另一个进程(wc -l)将标准输入重定向到管道文件,只有这样,操作系统才默认,进程(who)是向管道文件中拷贝数据而不是标准输出,另一个进程(wc -l)是向管道文件读取数据,而不是标准输入

进一步理解标准输出和标准输入,如上述的更改对应进程的标准输出和标准输入到管道文件,本质上都是根据原理,为了让管道文件成为传送数据那个进程(who)的标准输出,表示的就是将被传送的数据不拷贝到显示器上,而是拷贝到管道文件中,而另一个进程(wc -l),此时则不从标准输入(键盘)中获取数据,而是从管道文件中获取(重定向了)

上述这个知识点,困扰了我很久,原因可能是因为进程和文件描述符表之间的关系没玩明白,当然上述知识点需要贯穿理解的知识是有的,所以越往后学,需要掌握和运用的知识就越多,那么知识和知识之间的理解成本就会越高,所以基础一定要牢!并且上述这个知识点,可以说是理解管道最为重要的一点,搞定了这个,别的知识都只是顺水推舟摆了!具体看下图:

在这里插入图片描述

如上图所示,此时由于具有血缘关系的进程之间它们的文件描述符表都是继承于父进程,所以导致它们指向的文件是同一文件,并且由于操作系统维护了一个匿名管道文件,所以此时两个进程它们就共享一个管道文件,从而实现了让两个进程看到同一份资源,共享同一份资源的效果,最后由于每个进程对该共享资源都具有读写的方式,所以此时两个进程之间就可以通过控制读写,进而控制进程之间的通信例如,此时想要让子进程进行写入,父进程(也可能是兄弟进程)进行读取,此时要做的就是将子进程的读端关闭,父进程的写端关闭,具体是因为父进程和子进程对共享资源都具有读写的能力,但是由于管道只能进行单向传递(因为文件只有一个缓冲区),所以父进程和子进程不能对共享资源同时进行读写,只能你读我写或者是你写我读,只有满足这样的情况下,进程和进程之间才能成功完成通信(也就是该博客最开头的内容,进行资源共享,数据传递,通知事件,进程控制

并且从上述的血缘进程之间共享同一个个文件描述符表,可以解释一个现象,这个现象就是当时我们学习进程创建时,看到的,为什么创建子进程之后,printf或者cout,都是向标准输出(显示器)打印数据,原因就是因为,血缘进程之间共享文件描述符表(继承)

进程间通信代码实现

通过上述知识,有关进程间为什么要通信,如何通信等原理,我们都搞定了,并且具体的图我们也见过了,此时就让我们来完成学习进程间通信的最后一个过程,自己实现一个进程间通信的代码,如下所示:

想要自己实现一个进程间通信的代码,如上图所示,原理已经非常的明显了,具体就是分为如下几个步骤

1. 创建管道
2. 创建子进程
3. 关闭不需要的fd
4. 开始通信

所以此时就让我们按照上述步骤来自己实现一下进程间通信的代码吧!

创建管道
此时想要创建管道,如上述知识所说一般,管道本质上就是一个文件,并且这个文件是一个内存级文件,是有内核创建,然后操作系统进行管理的一个文件,所以明白,这个文件我们肯定是不可能创建出来的,想要创建出这个内存级文件,就必须要调用系统调用接口,让操作系统通过内核帮我们创建,此时这个系统调用接口:pipe

在Linux系统下使用说明如下:
在这里插入图片描述
如上图,pipe的基本使用说明,我们可以看出如下信息:

pipe函数用于创建一个管道,以实现进程间的通信,pipe函数的定义如下: 头文件#include<unistd.h> 函数接口调用int pipe(int pipefd[2]); pipe函数定义中的pipefd参数是一个大小为2的一个数组类型的指针该函数成功时返回0,并将一对打开的文件描述符值填入pipefd参数指向的数组,失败时返回 -1并设置错误码(errno)

代码如下:
在这里插入图片描述
所以此时通过上述的代码,此时我们就构建出了,如下图所示的管道(本质就是:在一个进程中,打开两个文件,一个以读的方式打开,一个以写的方式打开)
在这里插入图片描述

要注意的是,此时pipe接口参数是一个输出型参数,目的就是为了获取到被打开文件的文件描述符,以便于我们后序构建单向通信时,将进程之间不需要的文件描述符关闭,并且注意:pipe接口的参数是一个数组指针,由于数组天生就是以地址的形式出现,所以此时pipe接口参数是一个地址,也就是一个输出型参数,改变了pipe接口中该数组的值,该接口外部,该数组也会跟着改变,此时又因为pipe接口调用结束之后,会将两个打开文件的文件描述符拷贝给该数组,所以最终该接口调用成功之后,此时该数组的0下标和1下标表示的就是被打开文件的文件描述符,但此时因为我们要构建单向通信(管道要求),所以此时当创建完子进程之后(fork),就需要将进程不需要的fd关闭,又因为两个进程的文件描述符表是完全相同的,所以我们就可默认此时该数组0下标是以读方式打开的文件描述符,1下标是写方式打开的文件描述符,然后假如,此时是子进程向父进程传送数据,那么此时就将子进程的0下标,也就是以读方式打开的文件关闭,再将父进程的1下标关闭,也就是以写方式打开的文件关闭(反正本质就是将两个进程以读写方式打开的文件关闭一个,但必须是不同的那一个),这样一个单向通信的信道我们就构建出来了,具体如下图和代码所示:
在这里插入图片描述
此时根据上述代码,我们就可以得到如下图所示进程间通信的环境搭建了,此时根据下图原理,上图代码,此时进程间就可以进行数据传输等行为了
在这里插入图片描述

如下是我们接下来会学习的知识,这里由于时间关系,下篇博客搞定

System V IPC(本地通信标准)

明白了上述知识之后,此时我们就来认识一下System V进程间通信标注,其中包括了进程间通信的消息队列、共享内存、信号量等标准的相关知识!

消息队列

具体先看该链接:1.消息队列知识详解 2.消息队列相关知识理解 3.消息队列互补知识理解
该三个链接文章,足以让我们搞懂什么是消息队列了

共享内存

信号量

POSIX IPC(网络通信标准)

消息队列、共享内存、信号量、互斥量、条件变量、读写锁

北京时间:2023/4/13/7:49,早八人!润
在这里插入图片描述

总结:该篇博客本在昨天可以发的,可惜,由于自己的问题,vscode搞了半天,不过问题终于是解决了,所以问题不大,哈哈哈!并且发现,进程间通信是一个大知识点,尤其是进程的知识点本就多,现在加上通信,前路艰难呀!

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

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

相关文章

Linux下让进程不再被拉起

Linux下为了防止应用挂掉&#xff0c;我们会设置服务进程来拉起这些应用。但某些流氓软件也会利用该机制使得它们被杀掉后能再被拉起来。本文讲述让这些进程不再被拉起的方法。 比如&#xff0c;有名称为recordmain.bin的进程&#xff0c;使用kill -9 杀掉它后&#xff0c;过几…

NVIDIA jetson tensorrt加速yolov5摄像头检测

link 在使用摄像头直接检测目标时&#xff0c;检测的实时画面还是有点慢&#xff0c;下面是tensorrt加速过程记录。 一、设备 1、设备jetson agx xavier 2、jetpack4.6.1 3、tensorrt 8.2.1.8 4、conda虚拟环境 python3.6 二、虚拟环境搭建及依赖 1、参考此博客安装torch Nvidi…

做自动化测试时所谓的“难点”

这篇关于自动化测试的文章&#xff0c;可能和你看到的大多数自动化的文章有所不同。我不是一位专职的自动化测试工程师&#xff0c;没有开发过自动化的工具或者框架&#xff0c;用的自动化的工具也不多&#xff0c;也没有做过开发&#xff0c;所以我讲不出那些现在很多人很看重…

[C++]日期类计算器的模拟实现

目录 日期类计算器的模拟实现&#xff1a;&#xff1a; 1.获取某年某月的天数 2.构造函数 3.拷贝构造函数 4.赋值运算符重载 5.析构函数 6.日期天数 7.日期天数 8.日期-天数 9.日期-天数 10.前置的运算符重载 11.后置的运算符重载 12.前置--的运算符重载 13.后置--的运算符重载…

前后端交互系列之Axios详解(包括拦截器)

目录前言一&#xff0c;服务器的搭建二&#xff0c;Axios的基本使用2.1 Axios的介绍及页面配置2.2 如何安装2.3 Axios的前台代码2.4 Axios的基本使用2.5 axios请求响应结果的结构2.6 带参数的axios请求2.7 axios修改默认配置三&#xff0c;axios拦截器3.1 什么是拦截器3.2 拦截…

Go分布式爬虫笔记(二十)

文章目录20 调度引擎调度引擎目标通道函数选项模式函数式选项模式的好处通道底层原理无缓冲区的通道带缓冲区的通道Select 机制的底层原理思考题在我们的课程中&#xff0c;schedule 函数其实有一个 bug&#xff0c;您能看出来吗&#xff1f;你觉得可以用什么方式找出这样的 Bu…

OTA A/B 分区升级 update_engine简介

近期猛然发现公司的项目都已经换成了AB升级&#xff0c;AB升级之前一直有所了解&#xff0c;只是一直都没有去仔细查看过其具体升级流程&#xff0c;这两天抽空捋了捋&#xff0c;简单整理下。 AB升级&#xff08;谷歌官网叫法无缝更新&#xff09;是自android7.0开始新增的一…

头歌(Linux之进程管理一):第2关:进程创建操作-fork

任务描述 在上一关我们学习如何获取进程的pid信息&#xff0c;本关我们将介绍如何编程创建一个新的进程。 本关任务&#xff1a;学会使用C语言在Linux系统中使用fork系统调用创建一个新的进程。 相关知识 在Linux系统中创建进程有很多函数可以使用&#xff0c;其中包括了系…

初识Elasticsearch

文章目录介绍一、什么是elasticsearch&#xff1f;二、基本概念三、安装elasticsearch与kibana四、安装kibana&#xff08;跟ES要在同一个网络中&#xff09;五、IK分词器总结介绍 好处&#xff1a;可以帮助从海量数据中查找需要的内容&#xff1b; 一、什么是elasticsearch&…

ETL工具-pentaho企业实战部署

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

C++二叉搜索树与KV模型

二叉搜索树与KV模型二叉搜索树概念与操作性能分析实现KV模型二叉搜索树 本章是为了C的map和set做铺垫 概念与操作 二叉搜索树又称二叉排序树&#xff0c;它或者是一棵空树&#xff0c;或者是具有以下性质的二叉树: 若它的左子树不为空&#xff0c;则左子树上所有节点的值都小…

面试题之vue的响应式

文章目录前言一、响应式是什么&#xff1f;二、Object.defineProperty二、简单模拟vue三、深度监听四、监听数组总结前言 为了应对面试而进行的学习记录&#xff0c;可能不够有深度甚至有错误&#xff0c;还请各位谅解&#xff0c;并不吝赐教&#xff0c;共同进步。 一、响应式…

如何做好 IT 项目管理?做好项目管理常用的9大项目管理平台、7大管理方法

一个好的管理&#xff0c;是70%在流程、规范、工具&#xff0c;剩下的30%自由发挥。一个不好的管理&#xff0c;只有地板&#xff0c;每个人都要自己想办法&#xff0c;够到天花板。一个好的工具&#xff0c;就是帮助团队够到天花板的台阶。——刘润 项目管理是一门复杂的艺术&…

统一的文件管理,团队轻松协作

目前IT行业大都采用项目经理制的管理方式&#xff0c;这种管理方式下各个部门间相互独立&#xff0c;同时各部门间也缺乏沟通协作。因此IT行业在文件管理上主要面临以下几个问题&#xff1a; 文档缺乏集中管理&#xff1a;企业在管理过程中产生的大量文件分散在各个部门中&…

Python升级 pip : python -m pip install --upgrade pip,socket.timeout加入超时处理方法

人生苦短&#xff0c;我用python 最近又遇到了一个小的报错问题&#xff0c; 趁现在我还没有忘记&#xff0c; 赶紧来写一写… python 安装包资料报错交流:点击此处跳转文末名片获取 WARNING: You are using pip version 19.3.1; however, version 20.0.2 is available. You…

系统学习Numpy(一)——numpy的安装与基础入门[向量、矩阵]

系列文章目录 numpy的安装与基础入门[向量、矩阵与维度] numpy的安装与基础入门[向量、矩阵与维度]系列文章目录前言numpy安装向量与矩阵生成向量生成矩阵向量类型前言 numpy是科学计算以及机器学习深度学习的基础必备工具&#xff0c;本文将介绍numpy的安装&#xff0c;以及…

C语言课设项目-51单片机-中断系统

(创作不易&#xff0c;感谢有你&#xff0c;你的支持&#xff0c;就是我前行的最大动力&#xff0c;如果看完对你有帮助&#xff0c;请留下您的足迹&#xff09; 目录 51单片机的中断系统 一、中断的概念 二、51单片机的中断系统结构 三、中断允许控制 四、中断…

C#,初学琼林(06)——组合数的算法、数据溢出问题的解决方法及相关C#源代码

1 排列permutation 排列&#xff0c;一般地&#xff0c;从n个不同元素中取出m&#xff08;m≤n&#xff09;个元素&#xff0c;按照一定的顺序排成一列&#xff0c;叫做从n个元素中取出m个元素的一个排列(permutation)。特别地&#xff0c;当mn时&#xff0c;这个排列被称作全…

vs code c语言断点调试window版解决方案

序&#xff1a; 1、这一步不懂劝退多少人&#xff0c;博主搜到了多少博文都是mac的&#xff0c;结果发现都对不上&#xff01; 先看最终效果演示 接下去我每个步骤&#xff0c;你都仔细看&#xff0c;漏看一个环境都对不上&#xff01; 正文 1、先去看博主的c/c运行环境配置图…

10-vue3动画

文章目录1.vue的transition动画1.1transition的基本使用1.2transition组件的原理1.3过渡动画的class1.4class的命名规则和添加时机1.5显示的指定过渡时间1.6过渡的模式mode1.7动态组件的切换1.8.appear初次渲染2、animation动画2.1同时设置animation和transition3.结合第三方库…