【进程间通信】管道

  • (꒪ꇴ꒪ ),Hello我是祐言QAQ
  • 我的博客主页:C/C++语言,数据结构,Linux基础,ARM开发板,网络编程等领域UP🌍
  • 快上🚘,一起学习,让我们成为一个强大的攻城狮!
  • 送给自己和读者的一句鸡汤🤔:集中起来的意志可以击穿顽石!
  • 作者水平很有限,如果发现错误,请在评论区指正,感谢🙏

           进程间通信(IPC)是操作系统中用于不同进程之间交换数据和信息的一种机制。

        当谈到进程间通信时,管道(Pipe)是一种常见且简单的通信方式。管道主要用于在两个相关进程之间传递数据一个进程充当数据的发送者,而另一个进程充当数据的接收者。它也可以在父进程和子进程之间创建,或者在同一台机器上的不同进程之间创建。数据在管道中按照先进先出的顺序进行传递。

        管道有两种类型:无名管道(Anonymous Pipe)和命名管道(Named Pipe,也称为FIFO)。下面就让我们来详细了解一下这两种管道。

一、无名管道(Anonymous Pipe)

1.概述

        无名管道只能用于亲缘进程间(比如父子进程、兄弟进程、祖孙进程等)创建的一种通信方式写入操作不具有写入原子性,因此只能用于一对一的简单通信情形。无名管道只能在创建进程时使用,无法在其他无关进程之间使用。此外还具有以下特点:

        (1)单向性:数据只能从管道的写入端流向读取端;
        (2)缓冲区:无名管道通常具有有限的缓冲区,用于存储在写入时尚未读取的数据;
        (3)阻塞:如果管道已满(写入端写入速度过快),写入操作可能会阻塞,直到有足够的空间来容纳数据;
        (4)关闭:一旦所有引用管道的进程都关闭了管道,数据就会被清空;

        (5)没有名字,因此无法使用 open( )函数,也不能使用 lseek( )函数来定位。

还需要注意的是:无名管道的读写端是固定的。

        fd[0] :读端,只能进行读操作;

        fd[1]:写端,只能进行写操作。

2.创建无名管道

#include <unistd.h>
int pipe(int pipefd[2]);
参数:pipefd 一个至少具有 2 个 int 型数据的数组,用来存放 PIPE 的读写端描述符

3.示例

        创建一个子进程,实现键盘输入要发送的消息,父进程给子进程发消息,并且能够循环发送,子进程接收消息并显示。

        代码:

#include <stdio.h>
#include <unistd.h>
#include <string.h>int main(int argc, char const *argv[])
{// 创建一条无名管道int fd[2];int p = pipe(fd);  // 创建管道,fd[0] 是读端,fd[1] 是写端if (p == -1){perror("pipe fail");  // 输出错误信息return -1;}pid_t x = fork();  // 创建子进程if (x > 0)  // 父进程{char w_buf[100];while(1)  // 无限循环{printf("请输入要传输的字符串:");scanf("%s", w_buf);  // 从用户输入中读取字符串write(fd[1], w_buf, strlen(w_buf)+1);  // 将字符串写入管道的写端sleep(1);  // 等待1秒钟}}if (x == 0)  // 子进程{char r_buf[100];while(1)  // 无限循环{bzero(r_buf, sizeof(r_buf));  // 清空读缓冲区read(fd[0], r_buf, sizeof(r_buf));  // 从管道的读端读取数据到缓冲区printf("father say:%s\n", r_buf);  // 打印读取的数据}}return 0;
}

二、命名管道(Named Pipe / FIFO)

1.概述

         命名管道是一种更为通用的管道,又称有名管道,可以用于没有亲缘关系的进程之间的通信。与无名管道不同,命名管道可以在系统中存在一段时间,允许多个进程随时通过使用相同的管道名称来进行通信。其特点包括:

        (1)可命名性:通过指定一个唯一的名称(FIFO),进程可以随时打开并使用管道;
        (2)跨进程通信:不同进程可以通过打开相同名称的管道来进行通信;
        (3)非阻塞:命名管道通常可以设置为非阻塞模式,即使管道未就绪,进程仍然可以继续运行;
        (4)持久性:命名管道会保留在文件系统中,直到被明确删除;

        (5)跟普通文件一样可使用统一的 read( )/write( )来读写。任何具有相应权限的进程都可以使用 open( )函数来获取 FIFO 的文件描述符。

2.创建命名管道

#include <sys/types.h>
#include <sys/stat.h>
int mkfifo(const char *pathname, mode_t mode);
参数:
    pathname FIFO 的文件名
    mode 文件权限

3.示例

        设计两个进程,进程间通过有名管道进行通信,两个进程都既可以写数据,也可读数据,当写入的数据是“quit”的时候,写进程关闭;读进程收到“quit”,也关闭。

         完整代码:

jack.c

#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdlib.h>#define PATHFIFO1 "/tmp/myfifo1"
#define PATHFIFO2 "/tmp/myfifo2"int main(int argc, char const *argv[])
{//创建有名管道,如果已经存在,不创建就行了if (access(PATHFIFO1, F_OK)){int fifo = mkfifo(PATHFIFO1, 0777);if (fifo == -1){perror("mkfifo");return -1;}}if (access(PATHFIFO2, F_OK)){int fifo = mkfifo(PATHFIFO2, 0777);if (fifo == -1){perror("mkfifo");return -1;}}//打开有名管道int fifo_fd1 = open(PATHFIFO1, O_RDWR);if (fifo_fd1 == -1){perror("open");return -1;}int fifo_fd2 = open(PATHFIFO2, O_RDWR);if (fifo_fd2 == -1){perror("open");return -1;}pid_t x = fork();if (x > 0){char w_buf[100];while(1){scanf("%[^\n]", w_buf);//%[^\n]:除了'\n'这个字符,我全都要,慎用   %s:接收字符串的时候,不会接收空格while(getchar()!='\n');//发送数据write(fifo_fd1, w_buf, strlen(w_buf));if (strncmp(w_buf, "quit", 4) == 0){exit(0);}}}if (x == 0){char r_buf[100];while(1){bzero(r_buf, sizeof(r_buf));read(fifo_fd2, r_buf, sizeof(r_buf));printf("rose:%s\n", r_buf);if (strncmp(r_buf, "quit", 4) == 0){//自己发个信号杀死父进程return -1;}}}return 0;
}

rose.c

#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdlib.h>#define PATHFIFO1 "/tmp/myfifo1"
#define PATHFIFO2 "/tmp/myfifo2"int main(int argc, char const *argv[])
{//创建有名管道,如果已经存在,不创建就行了if (access(PATHFIFO1, F_OK)){int fifo = mkfifo(PATHFIFO1, 0777);if (fifo == -1){perror("mkfifo");return -1;}}if (access(PATHFIFO2, F_OK)){int fifo = mkfifo(PATHFIFO2, 0777);if (fifo == -1){perror("mkfifo");return -1;}}//打开有名管道int fifo_fd1 = open(PATHFIFO1, O_RDWR);if (fifo_fd1 == -1){perror("open");return -1;}int fifo_fd2 = open(PATHFIFO2, O_RDWR);if (fifo_fd2 == -1){perror("open");return -1;}pid_t x = fork();if (x > 0){char w_buf[100];while(1){scanf("%[^\n]", w_buf);//%[^\n]:除了'\n'这个字符,我全都要,慎用   %s:接收字符串的时候,不会接收空格while(getchar()!='\n');//发送数据write(fifo_fd2, w_buf, strlen(w_buf));if (strncmp(w_buf, "quit", 4) == 0){exit(0);}}}if (x == 0){char r_buf[100];while(1){bzero(r_buf, sizeof(r_buf));read(fifo_fd1, r_buf, sizeof(r_buf));printf("jack:%s\n", r_buf);if (strncmp(r_buf, "quit", 4) == 0){//自己发个信号杀死父进程return -1;;}}}return 0;
}

三、总结

        总的来说,管道是一种简单且有效的进程间通信方式,特别适合在具有亲缘关系的进程之间传递数据。但对于更复杂的通信需求,可能需要考虑其他IPC方式。常见的应用呢就是写入日志文件,我们该问题进行一个简化就可得到如下的图示:

        完成这个任务程序也很简单,我们需要一个总的日志文件保存程序和五个发送日子程序,这样我们就能模拟出日志文件的产出和写入,像这样:

        此外,在使用管道进行进程间通信需要注意以下几点:

        (1) 管道是基于字节流的,因此数据传输不会被分段;
        (2)管道的写入和读取操作是阻塞的,可能需要额外的机制来处理阻塞情况;
        (3) 管道通常用于在具有亲缘关系的进程(例如父子进程)之间进行通信,或者在需要共享数据的进程之间进行通信。

        更多C/C++语言Linux系统数据结构ARM板实战相关文章,关注专栏:

   手撕C语言

            玩转linux

                    脚踢数据结构

                            系统、网络编程

                                     探索C++

                                             6818(ARM)开发板实战

📢写在最后

  • 今天的分享就到这啦~
  • 觉得博主写的还不错的烦劳 一键三连喔~
  • 🎉🎉🎉感谢关注🎉🎉🎉

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

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

相关文章

【base64】JavaScriptuniapp 将图片转为base64并展示

Base64是一种用于编码二进制数据的方法&#xff0c;它将二进制数据转换为文本字符串。它的主要目的是在网络传输或存储过程中&#xff0c;通过将二进制数据转换为可打印字符的形式进行传输 JavaScript 压缩图片 <html><body><script src"https://code.j…

如果应对2023年国赛

国赛倒计时一周&#xff0c;大家多看看优秀论文&#xff0c;赛前多思考&#xff0c;使大脑在活跃状态&#xff0c;更好的应对题目。 需要历年游戏论文的小伙伴可私信我&#xff0c;或关注微信公众号私信我

每日一题(反转链表)

每日一题&#xff08;反转链表&#xff09; 206. 反转链表 - 力扣&#xff08;LeetCode&#xff09; 思路&#xff1a; 可以定义一个新的newhead结构体指针。再定义cur指针和next指针互相配合&#xff0c;将原链表中的节点从头到尾依次头插到newhead链表中&#xff0c;同时更…

Idea中使用Statistic插件统计工程项目代码量

1. 功能背景 公司要对一个项目进行代码统计&#xff0c;这么多类&#xff0c;总不能让我一个一个数据&#xff0c;于是想到了Statistic插件。让我们一起看看Statistic插件怎么使用吧。 2. Statistic插件 首先需要知道Idea统计项目代码行数&#xff0c;主要是使用Statistic插…

Vue的使用(2)

一个简单的Vue项目的创建 创建一个UserList.vue组件 在App.vue中使用该组件 使用组件的第一步&#xff0c;导入组件使用组件的第二部&#xff0c;申明这个组件使用组件的第三步&#xff1a;引用组件 结果&#xff1a; 事件和插值语法 <template> <div><!-- te…

深度学习1.卷积神经网络-CNN

目录 卷积神经网络 – CNN CNN 解决了什么问题&#xff1f; 需要处理的数据量太大 保留图像特征 人类的视觉原理 卷积神经网络-CNN 的基本原理 卷积——提取特征 池化层&#xff08;下采样&#xff09;——数据降维&#xff0c;避免过拟合 全连接层——输出结果 CNN …

Oracle DBlink使用方法

DBlink作用&#xff1a;在当前数据库中访问另一个数据库中的表中的数据 create public database link dblink名称 connect to 对方数据库用户名 identified by 对方数据库用户密码 using (DESCRIPTION (ADDRESS_LIST (ADDRESS (PROTOCOL TCP)(HOST 要连接的数据库所在服务…

系统学习Linux-keepalived

目录 keepalived双机热备 keepalivedlvs&#xff08;DR&#xff09; 1.实验环境 先配置主调度器 web节点配置 keepalived双机热备 web服务器安装nginx和keepalived 配置好这些可以进行漂移了 复制keepalived.conf 进行修改web1为主web2为从优先级设置好 #we…

智慧仲裁 | 祝贺「璞华法律自助咨询服务平台」在成都医学城劳动纠纷一站式联处中心正式上线运营!

利用互联网开展调解&#xff0c;已成为基层调解的常态。2023年8月&#xff0c;璞华劳动人事法律自助咨询服务平台在成都医学城劳动纠纷一站式联处中心正式上线运营&#xff01; 平台可以提供智能法律咨询、法律文书模版、赔偿计算、法律法规、仲裁指引、调解指引以及各种劳动人…

springmvc没有绿标,怎么配置tomcat插件运行?

一、添加插件后&#xff0c;刷新&#xff0c;自动从maven仓库下载tomcat插件 二、写好项目后&#xff0c;添加tomcat配置 三、即可点击绿标运行

煤矿皮带运输智能监控算法 opencv

煤矿皮带运输智能监控算法通过opencvpython深度学习算法网络模型&#xff0c;煤矿皮带运输智能监控算法实时监测皮带运输过程中的各种异常情况&#xff0c;如跑偏、撕裂、堆料异常等&#xff0c;一旦检测到异常情况&#xff0c;立即发出告警并采取相应的措施&#xff0c;以保障…

matlab使用教程(29)—微分方程实例

此示例说明如何使用 MATLAB 构造几种不同类型的微分方程并求解。MATLAB 提供了多种数值算法来求解各种微分方程&#xff1a; 1.初始值问题 vanderpoldemo 是用于定义 van der Pol 方程的函数 type vanderpoldemo function dydt vanderpoldemo(t,y,Mu) %VANDERPOLDEMO Defin…