P13 Linux进程间通信——管道

2023-12-10 22:12

本文主要是介绍P13 Linux进程间通信——管道,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言    

                  

🎬 个人主页:@ChenPi

🐻推荐专栏1: 《Linux C应用编程(概念类)_@ChenPi的博客-CSDN博客》✨✨✨ 

🔥 推荐专栏2: 《C++_@ChenPi的博客-CSDN博客》✨✨✨

🛸推荐专栏3: ​​​​​​《链表_@ChenPi的博客-CSDN博客 》 ✨✨✨
🌺本篇简介  : 这一章我们学习Linux进程间通信中的管道

在 Linux系统中,以进程为单位分配和管理资源。

由于保护的缘故,一个进程不能直接访问另一个进程的资源,也就是说,进程之间互相封闭。

但在一个复杂的应用系统中,通常会使用多个相关的进程来共同完成一项任务,因此要求进程之间必须能够互相通信,从而来共享资源和信息。

所以,一个操作系统内核必须提供进程间的通信机制(IPC)。

目前,Linux使用的进程间的通信方式主要有

管道(pipe)和有名管道(FIFO)

信号(signal)

消息队列

共享内存

信号量

套接字(socket)

在本节中,将介绍管道和信号。

01管道

直接说管道,通常指的是无名管道

特点:

  1. 它是半双工的(即数据只能在一个方向上流动),具有固定的读端和写端。
  2. 它只能用于具有亲缘关系的进程之间的通信(也是父子进程或者兄弟进程之间)。
  3. 它可以看成是一种特殊的文件,对于它的读写也可以使用普通的read、write 等函数。但是它不是普通的文件,并不属于其他任何文件系统,并且只存在于内存中。

其实也很好理解

我们要记得就是半双工的就可以,因为它不能同时读写,然后通常也只能用于父子进程中

它不属于普通文件,因为不属于 其他任何文件系统,且只存在于内存中。

1 #include <unistd.h>
2 int pipe(int fd[2]);    // 返回值:若成功返回0,失败返回-1

当一个管道建立时,它会创建两个文件描述符:fd[0]为读而打开,fd[1]为写而打开。如下图:

要关闭管道只需将这两个文件描述符关闭即可

 若要数据流从父进程流向子进程,则关闭父进程的读端(fd[0])与子进程的写端(fd[1]);反之,则可以使数据流从子进程流向父进程。

#include <stdio.h>
#include <unistd.h>int main(){char writebuf[]="I am father";char readbuf[32]={0};int fd[2]; //0为读,1为写pipe(fd);//创建管道pid_t pid = fork();//创建子进程switch (pid){case -1:perror("fork?");break;case 0:printf("this is child : %d\n",pid);close(fd[1]);int n_re = read(fd[0],readbuf,sizeof(readbuf));printf("read : %d byte from father :%s\n",n_re,readbuf);break;default:printf("this is father :%d\n",pid);close(fd[0]);int n_wr = write(fd[1],writebuf,sizeof(writebuf));printf("##########################################\n");break;}return 0;}

我们来看这段代码, 我们定义了两个数组,分别是readBuf和WriteBuf

还定义了一个整形的数组fd,这个就是用于管道读写操作的句柄了

然后pipe就是创建管道了

创建管道后

我们创建fork创建一个子进程

switch就是用区分父子进程执行线路的

如果不是很理解可以看一下我上两章的内容,关于fork系统调用的

可以看到我们在父进程线路中,我们close(fd[]),也就是关闭了读端了,然后写入writeBuf,也就是I am father 

子进程也是同理,区别就是子进程是关闭写端,然后读取内容的

我们编译执行一下

我们可以看到,子进程中打印了I am father这句话

所以说管道读写确实可以实现父子进程间的通信

02有名管道

FIFO,也称为命名管道,它是一种文件类型。

特点

  1. FIFO可以在无关的进程之间交换数据,与无名管道不同。
  2. FIFO有路径名与之相关联,它以一种特殊设备文件形式存在于文件系统中。

函数原型

 #include <sys/stat.h>// 返回值:成功返回0,出错返回-1int mkfifo(const char *pathname, mode_t mode);

其中的 mode 参数与open函数中的 mode 相同。

一旦创建了一个 FIFO,就可以用一般的文件I/O函数操作它。

当 open 一个FIFO时,是否设置非阻塞标志(O_NONBLOCK)的区别:

  1. 若没有指定O_NONBLOCK(默认),只读 open 要阻塞到某个其他进程为写而打开此 FIFO。类似的,只写 open 要阻塞到某个其他进程为读而打开它。
  2. 若指定了O_NONBLOCK,则只读 open 立即返回。而只写 open 将出错返回 -1 如果没有进程已经为读而打开该 FIFO,其errno置ENXIO。

FIFO的通信方式类似于在进程中使用文件来传输数据,只不过FIFO类型文件同时具有管道的特性。在数据读出时,FIFO管道中同时清除数据,并且“先进先出”。 

 readdemo.c

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>int main (){char rebuf[32]={0};if((mkfifo("./file",0600)==-1)){perror("why");}int fd = open("./file", O_RDONLY);int n_re = read(fd,rebuf,32);printf("read : %d byte \nbuf : %s ",n_re,rebuf);return 0;
}

writedemo.c

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
int main (){char buf[]="hello world";int fd = open("./file", O_WRONLY);int n_write = write(fd,buf,sizeof(buf));printf("write %d byte to read grep\nbuf : %s",n_write,buf);return 0;}

我们编译先运行 读端,可以看到目录下创建了一个file文件,而且黄色的,这就是管道文件了

然后读端目前阻塞在这里了

原因:若没有指定O_NONBLOCK(默认),只读 open 要阻塞到某个其他进程为写而打开此 FIFO。类似的,只写 open 要阻塞到某个其他进程为读而打开它。

 之后我们在运行写端,我们可以看到read端不阻塞了,接收到消息,然后打印就打印信息了

这就是管道简单使用了

这篇关于P13 Linux进程间通信——管道的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/478540

相关文章

linux-基础知识3

打包和压缩 zip 安装zip软件包 yum -y install zip unzip 压缩打包命令: zip -q -r -d -u 压缩包文件名 目录和文件名列表 -q:不显示命令执行过程-r:递归处理,打包各级子目录和文件-u:把文件增加/替换到压缩包中-d:从压缩包中删除指定的文件 解压:unzip 压缩包名 打包文件 把压缩包从服务器下载到本地 把压缩包上传到服务器(zip

Linux 网络编程 --- 应用层

一、自定义协议和序列化反序列化 代码: 序列化反序列化实现网络版本计算器 二、HTTP协议 1、谈两个简单的预备知识 https://www.baidu.com/ --- 域名 --- 域名解析 --- IP地址 http的端口号为80端口,https的端口号为443 url为统一资源定位符。CSDNhttps://mp.csdn.net/mp_blog/creation/editor

【Python编程】Linux创建虚拟环境并配置与notebook相连接

1.创建 使用 venv 创建虚拟环境。例如,在当前目录下创建一个名为 myenv 的虚拟环境: python3 -m venv myenv 2.激活 激活虚拟环境使其成为当前终端会话的活动环境。运行: source myenv/bin/activate 3.与notebook连接 在虚拟环境中,使用 pip 安装 Jupyter 和 ipykernel: pip instal

Linux_kernel驱动开发11

一、改回nfs方式挂载根文件系统         在产品将要上线之前,需要制作不同类型格式的根文件系统         在产品研发阶段,我们还是需要使用nfs的方式挂载根文件系统         优点:可以直接在上位机中修改文件系统内容,延长EMMC的寿命         【1】重启上位机nfs服务         sudo service nfs-kernel-server resta

【Linux 从基础到进阶】Ansible自动化运维工具使用

Ansible自动化运维工具使用 Ansible 是一款开源的自动化运维工具,采用无代理架构(agentless),基于 SSH 连接进行管理,具有简单易用、灵活强大、可扩展性高等特点。它广泛用于服务器管理、应用部署、配置管理等任务。本文将介绍 Ansible 的安装、基本使用方法及一些实际运维场景中的应用,旨在帮助运维人员快速上手并熟练运用 Ansible。 1. Ansible的核心概念

Linux服务器Java启动脚本

Linux服务器Java启动脚本 1、初版2、优化版本3、常用脚本仓库 本文章介绍了如何在Linux服务器上执行Java并启动jar包, 通常我们会使用nohup直接启动,但是还是需要手动停止然后再次启动, 那如何更优雅的在服务器上启动jar包呢,让我们一起探讨一下吧。 1、初版 第一个版本是常用的做法,直接使用nohup后台启动jar包, 并将日志输出到当前文件夹n

[Linux]:进程(下)

✨✨ 欢迎大家来到贝蒂大讲堂✨✨ 🎈🎈养成好习惯,先赞后看哦~🎈🎈 所属专栏:Linux学习 贝蒂的主页:Betty’s blog 1. 进程终止 1.1 进程退出的场景 进程退出只有以下三种情况: 代码运行完毕,结果正确。代码运行完毕,结果不正确。代码异常终止(进程崩溃)。 1.2 进程退出码 在编程中,我们通常认为main函数是代码的入口,但实际上它只是用户级

【Linux】应用层http协议

一、HTTP协议 1.1 简要介绍一下HTTP        我们在网络的应用层中可以自己定义协议,但是,已经有大佬定义了一些现成的,非常好用的应用层协议,供我们直接使用,HTTP(超文本传输协议)就是其中之一。        在互联网世界中,HTTP(超文本传输协议)是一个至关重要的协议,他定义了客户端(如浏览器)与服务器之间如何进行通信,以交换或者传输超文本(比如HTML文档)。

如何编写Linux PCIe设备驱动器 之二

如何编写Linux PCIe设备驱动器 之二 功能(capability)集功能(capability)APIs通过pci_bus_read_config完成功能存取功能APIs参数pos常量值PCI功能结构 PCI功能IDMSI功能电源功率管理功能 功能(capability)集 功能(capability)APIs int pcie_capability_read_wo