匿名管道的使用示例

news2025/1/13 5:49:51

目录

整体框架

 通信步骤

创建管道

​编辑创建子进程&关闭相应的fd

​编辑 

 进程间通信

父子进程通信之间四种场景

 实现父亲读,孩子写的进程间通信

管道通信的使用场景样例实现


整体框架

 通信步骤

创建管道

 

 pipe的参数为输出型参数,返回读写端的fd存储到pipefd的数组中,创建成功,函数返回值为0;否则为-1。

 

创建子进程&关闭相应的fd

 

 

 

 进程间通信

 

父子进程通信之间四种场景

1. 如果读端读取完管道中的所有数据,如果写端没有发送新的数据,该读端的进程只能等待;

2.如果写端写满了管道,而读端没有来的及去读,此时的写端无法继续去写

 

 

 

65535=2^16==2^12 * 2^4 == 4KB*16 

3. 如果关闭了写端,读取完毕管道数据,在读,read就会返回0,表示读到了文件的末尾。

4. 写端一直在写,读端关闭,会发生什么呢?写端在写就没有任何意义了。OS不会维护无意义、低效率、或者浪费资源的事情!OS会杀死一直在写的进程!OS会通过信号来终止进程,13)SIPIPE。

 实现父亲读,孩子写的进程间通信

//C++函数头文件c取代.h为了更好的兼容C++
#include<iostream>
#include<cerrno>
#include<cstring>
#include<cassert>
#include<string>

//系统调用函数的头文件为.h
#include<unistd.h>
#include<sys/types.h>
#include<sys/wait.h>

int main()
{
    //让不同的进程看到同一个资源 步骤:1~3
    //任何一种进程通信中,一定要 先 保证不同进程之间看到同一份资源  步骤:4

    int pipefd[2]={0};
    //1.创建管道
    int ret=pipe(pipefd);
    if(ret<0)
    {
        std::cout<<"pipe error:"<<errno<<":"<<strerror(errno)<<std::endl;
        return 1;
    }
    //验证管道是否创建成功
    std::cout<<"pipefd[0]:"<<pipefd[0]<<std::endl;//pipefd数组下标为0的表示读端
    std::cout<<"pipefd[1]:"<<pipefd[1]<<std::endl;//pipefd数组下标为1的表示写端


    //2.创建子进程
    pid_t id=fork();
    assert(id!=-1);

    if(id==0)
    {
        //子进程

        //3.关闭不需要的fd,让父进程进行读取,让子进程去写入
        close(pipefd[0]);

        //4.开始通信 ——结合某种场景
        // const std::string nameStr="Hello ,我是子进程";
        // int cnt=1;
        // char buffer[1024]={0};

        int cnt=0;

        while(true)
        {
            char x='X';
            write(pipefd[1],&x,1);
            std::cout << "Cnt: " << ++cnt<<std::endl;
            sleep(1);
            //break;

            // snprintf(buffer,sizeof(buffer),"%s,计数器:%d,我的pid:%d",nameStr.c_str(),cnt++,getpid());
            // write(pipefd[1],buffer,strlen(buffer));
            // sleep(10);
        }

        close(pipefd[1]);
        exit(0);
    }

    //父进程

    //3.关闭不需要的fd,让父进程进行读取,让子进程去写入
    close(pipefd[1]);

    //4.开始通信    ——结合某种场景
    char buffer[1024]={0};
    int cnt=0;
    while(true)
    {
        //sleep(5);
        int n=read(pipefd[0],buffer,sizeof(buffer)-1);
        if(n>0)
        {
            buffer[n]='\0';
            std::cout<<"我是父进程,child give me message:"<<buffer<<std::endl;
        }
        else if(n==0)
        {
            std::cout<<"我是父进程,读到了文件末尾"<<std::endl;
            break;
        }
        else 
        {
            std::cout<<"我是父进程,读取异常"<<std::endl;
            exit(1);
        }
        sleep(1);
        if(cnt++>5)break;
    }

    close(pipefd[0]);
    
    int status=0;
    waitpid(id,&status,0); 
    std::cout<<"sig:"<<(status&0x7F)<<std::endl;

    return 0;
}

管道通信的使用场景样例实现

【应用场景】

父进程要创建若干个子进程,每一个子进程配对一个管道,用来进行与父进程之间的通信;父亲通过管道,对子进程发送信号,子进程接受到信号后,依据信号来执行相应的任务!

【问题分析】

  1. 如何动态创建多个子进程?答:循环控制!
  2. 如何建立一个管道匹配一个子进程?答:再循环里面定义临时变量pipefd[2],这样就可以确保每一个创建的子进程找到对应读写的fd!
  3. 由于子进程会复制父进程的所有管理对象里的内容,这样,第二个以及后面的子进程就可以得到前面子进程与父进程管道之间读写fd信息,如何确保创建的管道只对应一个子进程,不受其它子进程的影响?答:因为,父进程只能关闭读端,不能关闭写端,所以要在父进程中保存之前子进程创建管道的写端的fd,后面的创建的子进程就可以得到之前子进程的读端的fd,在依次关闭就可以了!
  4. 父进程如何知道给哪一个子进程来分配任务?答:为了管理多个管道和子进程与父亲的关系,方法是先描述,在组织--》描述:父亲要得到子进程的fd才能找到相应的子进程,还要得到该子进程所对应管道写端的文件描述符fd--》组织:定义EndPoint结构体存储子进程的id和文件描述符fd;在最后,每创建一个子进程,就将相应的信息存储到vector<EndPoint>容器中。
  5. 如何指定相对应的任务呢?答:定义task对象,里面保存要有的任务,对每一个任务进行编号处理。用户通过输入数字来指定相对应的任务!

【代码实现思路】

  1. 首先需要创建子进程,并创建对应的管道;
  2. 进程间通信,输入对应执行任务指令;
  3. 进程退出,回收进程。

 代码实现:lesson14/2. CtrlProcess · 杰编程/LinuxCode - 码云 - 开源中国 (gitee.com)

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

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

相关文章

从小白到大神之路之学习运维第55天--------shell脚本实例应用

第三阶段基础 时 间&#xff1a;2023年7月7日 参加人&#xff1a;全班人员 内 容&#xff1a; shell实例 目录 shell脚本应用&#xff1a; 一、shell脚本 二、环境变量的基本使用 三、条件测试 shell脚本应用&#xff1a; 一、shell脚本 1、shell脚本的作用shell解…

MySql索引结构介绍

文章目录 1.索引概述2. 索引结构2.1 二叉树与红黑树索引结构2.2 B-Tree索引结构2.3 BTree介绍2.4 Hash索引介绍 1.索引概述 索引的概念 索引的演示 索引的优缺点 2. 索引结构 索引结构在不同的存储引擎中的支持情况&#xff0c;我们平时所说的索引&#xff0c;如果没有特殊的…

深度学习各类优化器大总结

一、优化算法设计原理 深度学习中的优化算法采用的原理是梯度下降法&#xff0c;即最小化目标函数 J ( θ ) J(\theta) J(θ)&#xff0c;最优化的求解过程&#xff0c;首先求解目标函数的梯度 ∇ J ( θ ) \nabla J(\theta) ∇J(θ)&#xff0c;然后将参数 θ \theta θ 向…

Java代码风格统一

Java代码风格统一 IDEA快捷键失效IDEA配置代码风格 IDEA快捷键失效 Ctr Shift F 这里是搜狗输入法影响的。 Ctr Shift L 这个是因为网易云的问题。 IDEA配置代码风格 统一代码风格是必须的&#xff0c;因为我们需要时不时的格式化代码&#xff0c; 如果每个人的开发风格…

程序员到公务员:从入门到上岸

有句话叫做&#xff0c;“打不过就加入”。尽管当下我们国家的体制还存在种种的问题&#xff0c;不过&#xff0c;大家还都是挤破脑袋想进体制。毕竟&#xff0c;某种程度上来说&#xff0c;体制内还真是香~ 我记得读大学那会&#xff0c;教我们C语言的老师&#xff0c;就和我…

Linus I/O之io_ring

由于调用系统调用时&#xff0c;会从用户态切换到内核态&#xff0c;从而进行上下文切换&#xff0c;而上下文切换会消耗一定的 CPU 时间。 io_uring 为了减少或者摒弃系统调用&#xff0c;采用了用户态与内核态 共享内存 的方式来通信。 第一步&#xff1a;应用程序通过向 io…

FreeRTOS ~(七)互斥量 ~ (2/3)互斥量解决优先级反转问题

前情提要 FreeRTOS ~&#xff08;四&#xff09;同步互斥与通信 ~ &#xff08;2/3&#xff09;互斥的缺陷 FreeRTOS ~&#xff08;五&#xff09;队列的常规使用 ~ &#xff08;2/5&#xff09;队列解决互斥缺陷 FreeRTOS ~&#xff08;六&#xff09;信号量 ~ &#xff08;2/…

openpnp - 用空调压缩机改真空泵

文章目录 openpnp - 用空调压缩机改真空泵概述笔记备注END openpnp - 用空调压缩机改真空泵 概述 原来设备用的真空泵是AP200C/V 这个商品标称是48DB的噪音, 实际用的时候, 噪音很大, 应该快70DB了, 放在屋子里很吵. 如果仅仅是太吵, 可能也忍了. 关键是这个真空泵有强干扰…

JS逆向之——天翼云登录

JS逆向之——天翼云登录 前言 此文章仅用于技术交流&#xff0c;欢迎指出相关问题&#xff0c;感兴趣的可以关注我&#xff0c;有相关python逆向问题可以私信交流讨论。 目标与目的 目标网站&#xff1a;天翼云登录 目的&#xff1a;通过构建登录参数&#xff0c;完成天翼…

Qt样式表Qss

文章目录 简介基本语法使用setStyleSheet设置样式表可以设置的属性使用QtCreator设置 简介 Qss是Qt程序界面中用来设置控件的背景图片、大小、字体颜色、字体类型、按钮状态变化等属性它是用来美化UI界面。实现界面和程序的分离&#xff0c;快速切换皮肤。 学习视频 基本语法…

网络编程4——TCP协议的三大安全机制:三次握手四次挥手+确认应答机制+超时重传机制

文章目录 前言一、TCP协议段与机制TCP协议的特点TCP报头结构TCP协议的机制与特性 二、TCP协议的 连接管理机制 TCP建立连接&#xff1a;三次握手 TCP断开连接&#xff1a;四次挥手 三、TCP协议的 确认应答机制 四、TCP协议的 超时重传机制 总结 前言 本人是一个刚刚上路的I…

unordered_map 浅析

【unordered_map 浅析】 ◆● unordered_maphttps://cplusplus.com/reference/unordered_map/ &#xff08;1&#xff09;unordered_map 适用于需要快速查找&#xff0c;且不要求元素存储顺序的场景&#xff0c;尤其适合使用自定义类型作为键的情况。但是需要注意的是&#xff…

RabbitMQ的集群

新建一个虚拟机,重新安装一个RabbitMQ,不会安装的可以看下面的连接: 在Linux中安装RabbitMQ_流殇꧂的博客-CSDN博客 1.修改/etc/hosts映射文件,两台虚拟机都需要修改 vim /etc/hosts 2.修改完成后重启linux服务器: reboot 3.查看.erlang.cookie文件的位置: find / -name .erlan…

环形链表 II

题目链接 环形链表 II 题目描述 注意点 如果 pos 是 -1&#xff0c;则在该链表中没有环 解答思路 首先判断链表中是否有环&#xff0c;思路为双指针当快慢指针相遇时&#xff0c;说明链表中有环&#xff0c;假设链表中非环的节点数量为a&#xff0c;链表中环的节点数量为b…

分享一个由rust实现的openai api服务端+Android客户端

开发缘由 官方网页存在经常中途断开的问题.经常使用不同ip登录openai帐号可能会导致封号.使用开源项目chatgpt-web搭建过一个网页端,目前已被DNS污染,体验GitHub Copilot.已经使用了rust语言一段时间,打算用它写个服务端练手. 服务端 技术栈 rust Rust是一种系统级编程语言…

Elasticsearch【域的属性、分词器、Elasticsearch搜索文档】(三)-全面详解(学习总结---从入门到深化)

目录 Elasticsearch常用操作_域的属性 分词器_默认分词器 分词器_IK分词器 分词器_拼音分词器 分词器_自定义分词器 Elasticsearch搜索文档_准备工作 Elasticsearch搜索文档_搜索方式 Elasticsearch常用操作_域的属性 index 该域是否创建索引。只有值设置为true&#…

【双指针】844. 比较含退格的字符串

844. 比较含退格的字符串 解题思路 思路&#xff1a;函数deLETE的作用是处理带有退格符号的字符串。它使用双指针法来模拟字符串处理。初始时&#xff0c;慢指针slow和快指针fast都指向字符串的开头。然后&#xff0c;通过遍历字符串&#xff0c;如果当前字符不是退格符号&…

postman报错提示 Could not get any response怎么解决

目录 postman报错 解决方法&#xff1a; 在通过postman请求做接口测试的过程中&#xff0c;有时候会遇到一些报错&#xff0c;当遇到这些报错我们不要着急&#xff0c;看着具体哪里报错&#xff0c;然后进行解决 postman报错 经常使用postman的小伙伴们都应该遇到过一些报错…

工作:三菱PLC之CC-LINK通讯知识及应用

工作&#xff1a;三菱PLC之CC-LINK通讯及应用 一. cc-link接线图&#xff08;RS485两线式接线&#xff09; CC-Link Ver.1.00兼容电缆&#xff08;特性阻抗&#xff1a;100Ω&#xff0c;终端电阻100Ω&#xff09; CC-Link Ver.1.10兼容电缆&#xff08;特性阻抗&#xff1a;…

Gradio库:使用Markdown模块创建交互式应用

❤️觉得内容不错的话&#xff0c;欢迎点赞收藏加关注&#x1f60a;&#x1f60a;&#x1f60a;&#xff0c;后续会继续输入更多优质内容❤️ &#x1f449;有问题欢迎大家加关注私戳或者评论&#xff08;包括但不限于NLP算法相关&#xff0c;linux学习相关&#xff0c;读研读博…