【Linux】匿名管道|命名管道|pipe|mkfifo|管道原理|通信分类|管道的特征和情况

news2025/1/23 21:59:06

目录

​编辑

进程间通信

为什么要有进程间通信

进程通信的目的

进程间通信分类

 如何理解通信

管道 

匿名管道

管道原理

半双工 

通信两问 

pipe 

演示

管道情况

管道的特征 

 命名管道

mkfifo指令 

mkfifo接口 

命名管道提供的是流式服务

匿名管道与命名管道的区别


 

进程间通信

进程间通信就是在不同进程之间传播或交换信息,进程间通信简称IPC(Interprocess communication);

通信的本质是让不同进程看到同一份资源

为什么要有进程间通信

在操作系统中,进程是独立运行的程序,它们之间需要相互协作完成任务。进程间通信的目的是为了实现进程之间的数据共享、协作和同步,从而提高系统的效率和可靠性。

进程通信的目的

  • 数据传输:一个进程需要将它的数据发送给另一个进程
  • 资源共享:多个进程之间共享同样的资源
  • 通知事件:一个进程需要向另一个或一组进程发送消息,通知它(它们)发生了某种事件(如进程终止时要通知父进程)
  • 进程控制:有些进程希望完全控制另一个进程的执行(如Debug进程),此时控制进程希望能够拦截另一个进程的所有陷入和异常,并能够及时知道它的状态改变

进程间通信分类

管道

  • 匿名管道
  • 命名管道)

System V IPC

  • System V 消息队列
  • System V 共享内存
  • System V 信号量

POSIX IPC

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

管道:管道是基于文件系统的,System V IPC:聚焦在本地通信,POSIX IPC:让通信可以跨主机 

 如何理解通信

  • 进程是相互独立的,想要达到两个进程之间的通信,这个必须要第三方来提供地方;
  • 操作系统提供了地点:公共资源;
    • 让进程看到这份公共资源,此时进程通信就有了前提;
  • 因为公共资源的种类不同,所以通信方式也不一样;
    • 如果公共资源是一块内存,那么通信方式就叫做共享内存;
    • 如果公共资源是一个文件,也就是struct file结构体,那么就叫做管道

管道 

管道一共有两种通信方案

        匿名管道和命名管道,底层原理基本是一样的,区别是它们各自的侧重点。 

匿名管道

匿名管道用于具有血缘关系的进程;常见于父子进程和兄弟进程;

匿名管道子进程继承父进程的文件描述符的内容来的

管道原理

让父进程以读和写的方式打开同一份文件,相当于一份文件被父进程打开了两次(只是为了好理解才这样表述,实际上创建管道,它有自己独立的接口);得到了两个不同的文件描述符;

子进程创建的时候,父进程的文件描述符表会被子进程继承下去,所以此时子进程在相同的fd处也指向父进程打开的文件。

文件描述符表一个进程维护一个,但是struct file结构体对象在内存中只有一个,由操作系统维护。

此时,父子进程将看到了同一份公共资源;就是操作系统在内存中维护的struct file对象,并且父子进程也都和这份资源建立了连接。

此时父子进程通信的基础有了,它们就可以通信了。

 

半双工 

管道只允许单向通信(半双工);

通信时,只能由一端写入另一端读,不能两端同时读或者写。这也就意味着,两个进程在通信时要有一个关闭读端,另一个关闭写端。这样才能保证通信稳定且有序的进行。
什么叫全双工呢?
简单来说,全双工就是通信时允许通信双方同时读或者同时写。

 

通信两问 

构建单向信道时父子进程最后都要关闭一个文件描述符,为什么曾经还要打开呢 ❓

因为如果父进程只以一种方式打开文件,那么fork()后的子进程都是对应着这种方式,这样会导致要么父子进程全是读要么全是写;

也有灵活运用,父子进程读写通信取决于场景,读写位置有时候会换;

对应的一组写和读可以不关闭吗 ❓

可以不关闭,但是防止絮乱,最好关闭(半双)

pipe 

创建匿名管道使用的系统调用;

成功返回0,失败返回-1

  • 形参:int pipefd[2]是一个输出型参数,是一个数组,该数组只有两个元素,下标分别为0和1。
  • 下标为0的元素表示的是管道读端的文件描述符fd。
  • 下标为1的元素表示的是管道写端的文件描述符fd。
int main()
{
    int pipefd[2];

    int ret =  pipe(pipefd);
    if(ret<0)
    {
        perror("pipe");
    }

    //创建子进程
   pid_t id = fork();
    if(0 == id)
    {
        //子进程 关闭读端
        close(pipefd[0]);
        
        //通信 TODO
       std::cout<<"child process close read!"<<std::endl;
        //子进程写端结束,关闭写端
        close(pipefd[1]);

        _exit(0);   //退出 清理 
    }

    //父进程
    close(pipefd[1]);   //父进程关闭写端
    
    //读取数据 TODO
    std::cout<<"father process close write!"<<std::endl;

    //读取完毕 关闭读端
    close(pipefd[0]);

    //等待子进程回收
        //父进程等待子进程结束
    int status = 0;
    pid_t pid = waitpid(id,&status,0);  //成功返回进程pid 失败返回-1
    std::cout<<"waitpid:"<<pid<<std::endl;
    sleep(3);
    
    return 0;
}

 

演示

void writer(int wfd)
{
    //写端
    char buff[1024];
    int count = 0;
    while(true)
    {
        //格式化消息,存放到buff缓冲区
        snprintf(buff,sizeof(buff),"pid:%d,count:%d",getpid(),count++);
        write(wfd,buff,strlen(buff));   //将缓冲区buff的数据写入管道

        sleep(1);
    }
}
void readr(int rfd)
{
    //读端
    char buff[1024];    //缓冲区
    while(true)
    {
        read(rfd,buff,sizeof(buff));    //在管道的读端 读取缓冲区的数据
        cout<<"get message:"<<buff<<endl;
    }
}

int main()
{
    int pipefd[2];
    int ret = pipe(pipefd);
    if(ret<0)
    {
        perror("pipe");
    }
    int rfd = pipefd[0];    //读端
    int wfd = pipefd[1];    //写端

    int id = fork();
    if(0 == id) //子进程
    {
        close(rfd); //读端   

        writer(wfd);

        _exit(0);
    }

    //父进程
    close(wfd);
    readr(rfd);

    waitpid(id,nullptr,0);  //等待子进程
    return 0;
}

 

管道情况

  • 如果当前管道里没有数据且写端没有被关闭。表示读取条件不具备,读端进程就会被阻塞,直到写进程写入数据。
  • 如果当前管道被写满且读端没有被关闭。表示写条件不具备,写端进程就会被阻塞,直到读进程读入数据。 (管道的空间是有限的,不同系统下的空间大小可能会有所区别。)
  • 管道一直在读,但是写端已经被关闭读端的read会返回0,表示读到了文件的末尾
  • 管道一直在写,但是读端已经被关闭。操作系统会像写端进程发送一个SIGPIPE(13)信号,强行关闭写端进程。(可以通过waitpid获取退出信号)管道也有大小,如果一次传输的数据太大且大于管道容量,linux将不保证写入的原子性。否则保证写入的原子性。

管道的特征 

  • 匿名管道只能用于具有血缘关系的进程之间的通信,通常使用于父子进程之间
  • 管道内部自带同步机制(读写操作的原子性)。这也是为什么我们的读端会在写端写入数据之后才会读取数据。这保证了数据在管道中流通的有序性。
  • 管道文件的生命周期随着进程的终止而结束。
  • 管道文件在通信时,是以字节流的方式读写数据。
  • 管道的通信模式是一种半双工模式。

 命名管道

匿名管道只能用于具有血缘关系的进程之间的通信,通常,一个管道由一个进程创建,然后该进程调用fork创建子进程,父子进程通过匿名管道进行通信。如果要实现两个毫不相关进程之间的通信,可以使用命名管道来做到;

  • 命名管道是一种特殊类型的文件 (FIFO 文件),命名管道有自己的路径 + 文件名。

mkfifo指令 

通过mkfifo指令来创建一个管道文件:

通过左侧第一个字符可知,这个文件的类型是p,也就是管道文件。 

向文件写入,此时命令行陷入了阻塞状态,等待别人读取这个管道的内容。

 我们在另外一个Bash中读取该管道 ;发现正常读取了;

mkfifo接口 

可以在进程中通过系统调用接口来创建管道mkfifo接口;

  • 返回0:创建成功;
  • 返回非0:创建失败

 

int mkfifo(const char* pathname, mode_t mode)
  • pathname:管道文件创建的路径
  • mode:管道文件的初始权限

 

如果要销毁这个管道文件,使用unlink接口

命名管道提供的是流式服务

  • 流式概念:提供一个通信的信道,写端就负责写,读端就负责读。但是,具体写多少、读多少完全由上层决定。底层就只提供一个数据通信的信道。它不关心数据本身的一些细节或格式,这叫做面向字节流。
  • 流式服务:数据没有明确的分割,一次拿多少数据都行

匿名管道与命名管道的区别

  1. 匿名管道由pipe函数创建并打开。
  2. 命名管道由mkfifo函数创建,打开用open
  3. FIFO(命名管道)与pipe(匿名管道)之间唯一的区别在它们创建与打开的方式不同,一但这些工作完成之后,它们具有相同的语义

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

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

相关文章

day08 1.进程间通信

work1.c #include <myhead.h> //要发送的消息类型 struct msgbuf {long mtype;char mtext[1024]; };#define SIZE sizeof(struct msgbuf)-sizeof(long)int main(int argc, const char *argv[]) {pid_t pid fork();if(pid -1){perror("fork error");return -…

Webpack入门基础知识及案例

webpack相信大家都已经不陌生了&#xff0c;应用程序的静态模块打包工具。前面我们总结了vue&#xff0c;react入门基础知识&#xff0c;也分别做了vue3的实战小案例&#xff0c;react的实战案例&#xff0c;那么我们如何使用webpack对项目进行模块化打包呢&#xff1f; 话不多…

RPA与智慧政务的关系

自1992年国务院明确提出构建全国行政机关办公决策系统&#xff0c;我国政府信息化建设已走过三十余年历程&#xff0c;并取得了阶段性成果&#xff0c;随着社会需求的变化以及信息技术和数字化工具的不断完善&#xff0c;人们对政府的信息化建设也提出了新的要求&#xff0c;推…

【C#语音文字互转】C#语音转文字(方法一)

Whisper.NET开源项目&#xff1a;https://github.com/sandrohanea/whisper.net/tree/main 一. 环境准备 在VS中安装 Whisper.net&#xff0c;在NuGet包管理器控制台中运行以下命令&#xff1a; Install-Package Whisper.net Install-Package Whisper.net.Runtime其中运行时包…

uniapp 实现自定义缩略滚动条

<template><view class"container-scroll"><!-- 文字导航 --><scroll-view class"scroll-view-text" scroll-x"true" v-if"type 1"><navigator:url"item.url"class"scroll-view-item"…

LE-50821F/FA激光扫描传感器|360°避障雷达之功能与连接使用说明

LE-50821F/FA激光扫描传感器|360避障雷达广泛应用于工业自动化、移动机器人应用场景中的环境感知、高精度定位&#xff08;如建图、扫描、避障、防护等&#xff09; LE-50xxxF系列升级扫描频率最高可达600KHz​​​​。 本文重点介绍LE-50821F/FA激光扫描传感器|360避障雷达之…

【C++】二维数组 数组名

二维数组名用途 1、查看所占内存空间 2、查看二维数组首地址 针对第一种用途&#xff0c;还可以计算数组有多少行、多少列、多少元素 针对第二种用途&#xff0c;数组元素、行数、列数都是连续的&#xff0c;且相差地址是有规律的 下面是一个实例 #include<iostream&g…

FreeRTOS基础入门——FreeRTOS的系统配置(三)

个人名片&#xff1a; ​ &#x1f393;作者简介&#xff1a;嵌入式领域优质创作者&#x1f310;个人主页&#xff1a;妄北y &#x1f4de;个人QQ&#xff1a;2061314755 &#x1f48c;个人邮箱&#xff1a;[mailto:2061314755qq.com] &#x1f4f1;个人微信&#xff1a;Vir202…

基于大模型的Agent

2023年&#xff0c;对于所有的人工智能领域只有一个共同的主题——大模型。大模型的受关注程度与发展速度可谓前所未有。其中&#xff0c;基于大模型的Agent又是最近几个月大模型领域的热点。这不开始研究没有几个月&#xff0c;综述文章都出来了&#xff0c;你说快不快&#x…

FashionAI比赛-服饰属性标签识别比赛赛后总结(来自 Top14 Team)

关联比赛: FashionAI全球挑战赛—服饰属性标签识别 推荐大家看本篇博客之前&#xff0c;看一下数据集制作的方法&#xff0c;如何做一个实用的图像数据集 PS&#xff1a;我是参加完比赛之后才看的&#xff0c;看完之后&#xff0c;万马奔腾.....&#xff0c;因为发现比赛中还…

62 函数参数——传递参数时的序列解包

与可变长度的参数相反&#xff0c;这里的序列解包是指实参&#xff0c;同样也有 * 和 ** 两种形式。 ① 调用含有多个位置参数的函数时&#xff0c;可以使用 Python 列表、元组、集合、字典以及其他可迭代对象作为实参&#xff0c;并在实参名称前加一个星号&#xff0c;Python …

element-ui/plus使用el-date-picker周 选择器返回时间范围处理案例

element-ui/plus使用el-date-picker周 选择器返回时间范围处理案例 如图所示 <el-date-pickerchange"changeTime":picker-options"{ firstDayOfWeek: 1 }"v-model"value1"type"week"format"YYYY年 第ww周"placeholder&…

C++初学者指南-5.标准库(第二部分)--数值运算算法

C初学者指南-5.标准库(第二部分)–数值运算算法 文章目录 C初学者指南-5.标准库(第二部分)--数值运算算法iota (注意不是itoa函数)Reductions reduce transform_reduce遗留操作&#xff08;无法并行执行&#xff09;accumulate (≈ reduce) C98inner_product (≈ transform_r…

sanger序列拼接--一次错误示范

文章目录 目的实现步骤 目的 NGS得到了很多的reads&#xff0c;其中有一些paired reads我想根据overlap 搭建起来&#xff0c;因为我对序列的ID做了删减&#xff0c;所以再pandaseq那里跑不通。 总结来说&#xff0c;目的很简单&#xff0c;就是把 有重叠区域的 reads 搭起来…

【学习笔记】A2X通信的协议(二)- A2X配置参数

目录 5. A2X配置参数 5.1 一般说明 5.2 A2X配置参数的配置和优先级 5.2.1 一般说明 5.2.2 A2X配置参数的优先级 5.2.3 通过PC5进行的A2X通信的配置参数 5.2.4 广播远程ID&#xff08;BRID&#xff09;的配置参数 5.2.5 直接检测和避免&#xff08;DDAA&#xff09;的配…

解决 Beyond Compare 30天过期问题

解决 Beyond Compare 30天过期的步骤如下&#xff1a; 1、使用快捷键WinR打开运行窗口&#xff0c;输入regedit并回车&#xff0c;打开注册表编辑器。 2、在注册表编辑器中&#xff0c;找到Beyond Compare的注册表位置&#xff0c;路径通常是HKEY_CURRENT_USER\Software\Scoot…

【redis】springboot 用redis stream实现MQ消息队列 考虑异常ack重试场景

redis stream是redis5引入的特性&#xff0c;一定程度上借鉴了kafka等MQ的设计&#xff0c;部署的redis版本必须 > 5 本文主要讲的是思路&#xff0c;结合简单的源码分析&#xff08;放心&#xff0c;无需深入大量源码&#xff09;&#xff1b;讲述在redis stream文档缺乏&a…

港科夜闻 | 香港科大,中大,港大及国大获旭日慈善基金捐款港币五千万元,支持基础数学研究及人才发展...

关注并星标 每周阅读港科夜闻 建立新视野 开启新思维 1、香港科大、中大、港大及国大获旭日慈善基金捐款港币五千万元&#xff0c;支持基础数学研究及人才发展。香港科大校长叶玉如教授在会上代表四所大学&#xff0c;就旭日慈善基金会对推动高等教育及基础研究发展的慷慨支持&…

探索智谱AI的视频生成神器:CogVideoX完全指南

引言 在当今数字化和内容创作高度发达的时代&#xff0c;视频已经成为信息传播和营销的重要工具。然而&#xff0c;对于许多缺乏视频制作经验或资源的个人和企业而言&#xff0c;如何快速、高效地创建吸引人的视频仍然是一个挑战。智谱AI推出的CogVideoX&#xff0c;作为一款先…

PuerTS和HybridCLR哪个更适合开发微信小游戏

1&#xff09;PuerTS和HybridCLR哪个更适合开发微信小游戏 2&#xff09;使用了Play Asset Delivery提交版本被Google报错 3&#xff09;怎样设置normalize来改变摄像机位置 4&#xff09;如何禁用增强型输入法中除某些输入操作之外的输入操作 这是第397篇UWA技术知识分享的推送…