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

进程通信(管道,共享内存实现)

01. 进程通信简介

进程通信工具分为数据传输工具和共享内存两类。这里我们讨论进程通信工具(IPC)里面的管道、system V和共享内存。在理解阶层通信之间,我们先了解用户空间缓冲区和内核空间缓冲区两个概念。

在这里插入图片描述

1.1 用户空间缓冲区

存在于用户态的进程用户空间,进程直接读取。用于存储待发送或已接收的数据,减少频繁的系统调用开销。

代码示例:

char buf[1024];
write(fd, buf, sizeof(buf));  // 将用户缓冲区buf数据写到fd指向内核
read(fd, buf, sizeof(buf));  // 从内核读取数据到用户缓冲区

**PS:**可以使用 fnprintf 先写入用户缓冲区,满后再通过 write 系统调用写入内核(后面有提到)。


1.2 内核空间缓冲区

存在于内核态作为管道通信的中转站,操作系统进行管理,使用系统调用访问


02. 管道(pipe)

我们再使用Linuxls | wc -l命令时,会用到 |这个符号表示的即是一个管道。为了执行这么命令创建了两个进程分别执行lswc

在这里插入图片描述

2.1 管道特点

为了通信,一个进程从用户内存向内核内存(进程通信)工具中写入数据,另外一个进程从内核内存读取数据到用户内存,这样便完成了数据的写入和读取,两个过程不能同时进行,且有以下要求:

  • 血缘关系:仅适用于有共同祖先(血缘关系)的进程;通常,一个管道由一个进程创建,然后该进程调用fork(),此后父、子进程之间就可应用该管道(下方有涉及)
  • 内核缓冲区:数据通过内核缓冲区传输,默认大小通常为4KB
  • 阻塞机制:当管道为空时,读操作阻塞;当管道满时,写操作阻塞;进程退出,管道释放
  • 半双工:数据只能单向流动若需双向通信,需创建两个管道

在这里插入图片描述

用户进程 用户缓冲区 内核管道缓冲区 准备数据(如 snprintf) write(fd, buf, sizeof(buf)) 返回实际写入字节数 read(fd, buf, sizeof(buf)) 填充数据 处理数据(如 printf) 用户进程 用户缓冲区 内核管道缓冲区

2.2管道的创建和使用

int pipe(int pipefds[2])//成功返回0,失败返回-1并且errno报错

pipe()系统调用创建一个新管道。成功的pipe()调用会在pipefds数组中返回两个打开的文件描述符:pipefds[0]表示读取端,pipefds[1]表示写入端。可以使用 read()和 write()系统调用来在管道上执行 I/O。一旦向管道写入数据之后立即就能从管道的读取端读取数据。但当管道为空时阻塞read()调用读取的是min{请求的字节数,当前管道存在字节数}。

在这里插入图片描述

2.2.1 匿名管道

父子进程通信过程:

  1. 父进程创建管道,得到两个⽂件描述符指向管道的两端

  2. 父进程fork出子进程,⼦进程也有两个⽂件描述符指向同⼀管道

  3. 子进程关闭pipefds[0]读端,父进程关闭pipefds[1]写端即可实现了进程间通信

#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <assert.h>
int main()
{int pipefds[2] = {0};int n = pipe(pipefds);assert(n == 0)// 0:读取端  1:写入端pid_t id = fork();assert(id >= 0) ;if (id == 0){                      // 子进程写入close(pipefds[0]); // 子进程关闭读端const char *msg = "hello Linux!";int cnt = 0;while (1){char buf[1024];fnprintf(buf, sizeof buf, "child is sauing:%s[%d] ", msg, cnt);write(pipefds[1], buf, strlen(buf));sleep(1);}exit(0);}// 父进程读取close(pipefds[1]); // 父进程关闭写端while (1){char buf[1024];ssize_t s = read(pipefds[0], buf, sizeof(buf) - 1); // 保证从二进制文件读取到用户进程某位空一个\0if (s == 0){ // 意味着子进程退出了printf("child quit\n");break;}else if (s > 0){buf[s] = 0;printf("child say to father# %s\n", buffer);}else{printf("read error\n");break;}}return 0;
}

加载中...
父子进程读取和写入分别在文件描述符中fd[3]fd[4],然后子进程为了写入关闭fd[3],父进程为了读取则需要关闭fd[4]

2.2.2 命名管道

system V共享内存

即允许一个进程将数据放到进程共享内存块中让其他进程读取这些数据来完成信息交换,此过程不涉及变换状态(内核态和用户态切换),使其速度非常快。

system V共享内存

即允许一个进程将数据放到进程共享内存块中让其他进程读取这些数据来完成信息交换,此过程不涉及变换状态(内核态和用户态切换),使其速度非常快。

http://www.lryc.cn/news/2386756.html

相关文章:

  • 电池预测 | 第28讲 基于CNN-GRU的锂电池剩余寿命预测
  • 快速上手SHELL脚本常用命令
  • 【无标题】前端如何实现分页?
  • 【自然语言处理与大模型】大模型Agent四大的组件
  • 小巧高效的目录索引生成软件
  • 云原生架构设计相关原则
  • android实现使用RecyclerView详细
  • 华为云Flexus+DeepSeek征文 | Flexus X实例助力 Dify-LLM 一键部署:性能跃升与成本优化的革新实践
  • 曼昆经济学原理第九版目录
  • 数据库blog7_MySql的下载与配置准备
  • YOLOv11助力地铁机场安检!!!一键识别刀具
  • RFID工业读写器的场景化应用选型指南
  • java中的线程安全的集合
  • 单片机如何快速实现查看实时数据
  • go实现钉钉三方登录
  • YOLOv1 详解:单阶段目标检测算法的里程碑
  • 5G 核心网切换机制全解析:XN、N2 与移动性注册对比
  • 物流配送优化实战:用遗传算法破解选址难题
  • Linux 个人用户设置账号密码环境变量,四种方式
  • Three.js搭建小米SU7三维汽车实战(5)su7登场
  • 关于 SSE(Server-Sent Events)过程的简要解剖
  • 格恩朗管段超声波流量计:流量测量先锋
  • 重构开发范式!飞算JavaAI革新Spring Cloud分布式系统开发
  • 图论 判断是否有环
  • (请关注)Oracle性能调优、优化总结调优参考直接应用,性能提升实用案例
  • EasyDarwin的配置与使用
  • PostgreSQL日志管理完整方案(AI)
  • 【Android】基于SurfaceControlViewHost实现跨进程渲染
  • vue+ThreeJs 创造自动选择的甜甜圈(圆环)
  • 能说一下JVM的内存区域吗