进程间通信之匿名管道和命名管道

news2025/7/12 20:37:07

目录

管道是什么

匿名管道

命名管道

命名管道创建方式

管道的特点:


管道是什么

概念管道是计算机通信领域设计者,设计出的一种单向通信的方式,linux原生提供管道通信

管道都是单向传输内容的
管道中传输的都是"资源"
管道通信不会将内容刷新到磁盘,是内存级的通信(效率高)

管道是半双工的:

全双工:既可以收又可以发,同时进行

半双工:要么在收要么发

匿名管道

//头文件
#include <unistd.h>

//函数
int pipe(int pipefd[2]);

//返回值
//成功返回0,失败返回-1,会设置错误码到errno中

//参数
//int pipefd[2]   
//输出型参数,我们提供一个数组传入,这个数组会被填充两个文件描述符,pipefd[0]是读,pipefd[1]是写

1、分别以读写方式打开同一个文件

2、fork()创建子进程

3、双方进程各自关闭自己不需要的文件描述符

4、进行通信

5、通信结束

#include<iostream>
#include<unistd.h>
#include<stdio.h>
#include<sys/types.h>
#include<sys/wait.h>
#include<string.h>
using namespace std;

int main()
{
    int pipefd[2];  //创建一个数组,接受文件描述符
    if(pipe(pipefd) < 0)    //判断管道是否创建成功
    {
        perror("pipe"); //失败打印pipe,然后退出
        return -1;
    }

    pid_t id = fork();  //创建一个进程
    if(id < 0)  //判读进程是否创建成功
    {
        perror("fork"); //失败打印fork,然后退出
        return -1;
    }

    if(id == 0) //我们让子进程进行读取
    {
        close(pipefd[1]);   //关闭写进程
        char buff[64];
        while(1)    //进行通信
        {
            ssize_t s = read(pipefd[0], buff, sizeof(buff));
            if(s > 0)
            {
                if(strcmp(buff, "q") == 0)
                {
                    cout << "quit" << endl;
                    break;
                }
                cout << "parent say: " << buff << endl;
            }
            else if(s == 0)
            {
                cout << "eof" << endl;
                break;
            }
            else
            {
                cout << "read error" << endl;
                break;
            }
        }
        close(pipefd[0]);   //关闭读进程
    }
    //父进程进程通信
    close(pipefd[0]);   //关闭读
    if(id > 0)
    {
        char buff[64];
        while(1)
        {
            cin >> buff;
            write(pipefd[1], buff, sizeof(buff));
            if(strcmp(buff, "q") == 0)
            {
                break;
            }
        }
    }

    //通信结束
    wait(NULL);
    close(pipefd[1]);   //关闭写
    return 0;
}

 

匿名管道的特点:

管道是用来进行具有血缘关系的进程进行进程间同通信 -- 常用于父子通信

a、写快,读慢,写满就不能写了

b、写慢,读快,管道没有数据的时候,读必须等待

c、写关,读0,表示读到了文件结尾

d、读关,写继续写,os终止写进程

命名管道

为了解决匿名管道只能父子间通信,产生了命名管道

命名管道创建方式

方式一:

mkfifo指令:

mkfifo 管道名        --创建一个命名管道

方式二:

mkfifo函数

//头文件
#include <sys/types.h>
#include <sys/stat.h>
//函数
int mkfifo(const char *pathname, mode_t mode);
//返回值    创建成功返回0,失败返回-1

//参数
//const char *pathname,创建的路径
//mode_t mode,创建权限

命名管道fifo是p标识管道文件

1、分别写一个客户端,一个服务端

2、服务端创建命名管道

3、服务端发送信息,客服端接受信息

4、通信结束,客服端和服务端退出

 服务端,server.cpp

#include<iostream>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<unistd.h>
#include<stdio.h>
#include<string.h>
using namespace std;

int main()
{
    umask(0000);    //将权限掩码设置为0000
    if(mkfifo("./fifo", 0666))  //创建命名管道fifo
    {
        perror("mkfifo");
        return -1;
    }

    int fd = open("./fifo", O_WRONLY); //以写方式打开命名管道
    if(fd < 0)
    {
        perror("open");
        return -1;
    }

    char buff[64];
    while(1)
    {
        cin >> buff;
        write(fd, buff, sizeof(buff));
        if(strcmp(buff, "q") == 0)
        {
            cout << "quit" << endl;
            break;
        }
    }
    unlink("./fifo");   //删除命名管道
    return 0;
}

 客服端,client.cpp

#include<iostream>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<unistd.h>
#include<stdio.h>
#include<string.h>
using namespace std;

int main()
{
    umask(0000);    //将权限掩码设置为0000
    if(mkfifo("./fifo", 0666))  //创建命名管道fifo
    {
        perror("mkfifo");
        return -1;
    }

    int fd = open("./fifo", O_WRONLY); //以写方式打开命名管道
    if(fd < 0)
    {
        perror("open");
        return -1;
    }

    char buff[64];
    while(1)
    {
        cin >> buff;
        write(fd, buff, sizeof(buff));
        if(strcmp(buff, "q") == 0)
        {
            cout << "quit" << endl;
            break;
        }
    }
    unlink("./fifo");   //删除命名管道
    return 0;
}

 

管道的特点:

1、匿名管道是用来进行具有血缘关系的进程进行进程间同通信 -- 常用于父子通信

2、管道具有通过让进程间协同,提供了访问控制 假如写满,需要等待被读取才能继续写入

3、管道提供的是面向流式的服务 -- 面向字节流 -- 协议

4、管道是基于文件的,文件的生命周期是随进程的,管道的生命周期是随子进程的

5、管道是单向通信的(半双工通信的一种特殊情况)

全双工:既可以收又可以发,同时进行

半双工:要么在收要么发

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

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

相关文章

SSM_整合篇

一、整合步骤 1.1 数据表的介绍 team表&#xff1a; player表&#xff1a; 1.2 创建maven项目 1.3 pom.xml引入jar依赖 <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0"xmlns:xsi"…

ps打开图片的三种方式 同步部分基本操作方式

观看本文 需要您的电脑已安装PS工具 如果没有 可以观看我的文章 PS软件下载安装以基本配置 然后打开PS 就会变成一个这样的界面 然后点击右上角的 PS 进入工作区 然后我们就会进入 一个这样的工作区 然后我们在左上角点击文件 选择 打开 然后 在文件框中 找到自己想处理的图…

OJ练习第23题——Z字形变换

OJ练习第23题——Z字形变换题目要求示例Java代码如下&#xff1a;思路分析力扣链接&#xff1a;Z字形变换题目要求 将一个给定字符串 s 根据给定的行数 numRows &#xff0c;以从上往下、从左到右进行 Z 字形排列。请你实现这个将字符串进行指定行数变换的函数&#xff1a;str…

canvas学习

canvas 是一块画布&#xff0c;可以设置宽高 &#xff0c;默认 300 * 150 使用方式 1. 声明书写 canvas标签 2. 拿到canvas的dom 3. 调用方法 getContext (注意 此方法在prototype上) 方法集合&#xff1a; 填充&#xff1a; 1. fillStyle&#xff0c; 设置填充颜色 &…

详解非负矩阵分解(NMF)及其在脑科学中的应用

非负矩阵分解及其在脑科学中的应用 基本原理确定最优因子数量代码实现非负矩阵分解与主成分分析的区别非负矩阵分解在脑科学中的应用应用一:神经发育模式:T2w/T1w比值映射的非负矩阵分解(NMF)应用二:微观结构的协方差模式基本原理 NMF的基本思想可以简单描述为:对于任意给…

Python用PyMC3实现贝叶斯线性回归模型

在本文中&#xff0c;我们将在贝叶斯框架中引入回归建模&#xff0c;并使用PyMC3 MCMC库进行推理。 最近我们被客户要求撰写关于叶斯线性回归模型的研究报告&#xff0c;包括一些图形和统计输出。我们将首先回顾经典频率论的多重线性回归方法。然后讨论贝叶斯如何考虑线性回归。…

8、MyBatis核心配置文件之typeAliases(mybatis-config.xml)

MyBatis核心配置文件之typeAliases&#xff08;mybatis-config.xml&#xff09; 1、&#xff01;&#xff01;&#xff01;&#xff01;注意 2、 设置类型别名&#xff08;比如有的全类名&#xff08;resultType&#xff09;太长了不好使用&#xff09; typeAlias :设置某个类…

Python版本的温湿度+Nokia5110 display(SPI)

前提需要把micropython的固件安装到系统中 安装micropython到esp8266中 本实验需要&#xff1a; 1. ESP8266&#xff08;我的是Wemos D1) 2. DHT11 3. Nokia5110 LCD 连线&#xff1a; DHT11 out --> D2(GPIO-016) (-接入GND&#xff0c;接入3.3vcc) Nokia 5110 LCD We…

GO语言最常用的语法

一 ,变量&#xff1a;变量赋值只能在函数内使用&#xff0c;故第三种方式只能在函数内使用&#xff0c;可使用var()同时定义多个变量变量定义 使用var关键字 var a bool var a bool true 不指定类型直接初始化让编译器选择 var a "abc" 使用 “ : "…

Python入门、环境搭建、变量、数据类型

目录 前景 官方下载 基本数据类型 动态语言的体现 静态语言的体现 弱语言的体现 强语言的体现 注释 整数 浮点型 浮点型计算方案 字符串 布尔 引用数据类型 列表 [ ] 列表方法 集合Set{} 基本方法 特殊需求方法 应用场景 字典{} 常见操作 元组 操作符 练习…

基于ANSYS 2019R1全解一款双吸泵的双向流固耦合方法

作者&#xff1a;李雷 一、导读 对于旋转机械来说&#xff0c;传统设计从理论计算到手工木模图&#xff0c;再到模型泵的加工制造&#xff0c;最后进行相关性能试验。当性能试验与预期效果差距较大的时候还需要修改水力模型。这种传统的设计不仅设计周期长&#xff0c;而且成…

Vue3+nodejs全栈项目(资金管理系统)——后端篇(二)用户模块

文章目录用户模块的增删改查新增创建user_info表初始化路由模块路由模块处理函数(添加&#xff09;测试查询路由模块处理函数(查询)测试编辑&#xff08;根据id&#xff09;路由模块处理函数&#xff08;编辑/更新&#xff09;测试删除(根据id&#xff09;路由模块处理函数测试…

黑马JVM学习笔记-内存结构

什么是JVM? 定义&#xff1a; Java Virtual Machine - java 程序的运行环境(Java二进制字节码的运行环境) 好处&#xff1a;3 一次编写&#xff0c;到处运行自动内存管理&#xff0c;垃圾回收功能数组下标越界检查(下标越界抛出异常比数组新元素覆盖其他部分造成的危害小)…

1. SAP Business Application Studio 里创建一个基于 CAP 模型的最简单的 OData 服务

本教程已经花费了 24 个文章的篇幅,介绍了使用 SAP ABAP SEGW 这个开发工具,开发基于 SAP ABAP 技术栈的 OData 服务的详细步骤。 正如本教程目录 中提到的那样,SAP OData 开发技术包含传统的 ABAP,RAP(Restful ABAP Programming) 和 CAP(Cloud Application Programming) …

前端程序员接私活,直呼赚麻了

总有一些前端程序员会想找私活&#xff0c;但是又不清楚具体的办法&#xff0c;或者是做了但没完全做&#xff0c;吃力又不讨好还赚不到钱。今天就给大家介绍一些可行性高的方法&#xff0c;让你快速找到合适的前端兼职。 干货满满&#xff0c;希望大家点赞收藏下&#xff0c;别…

Java 异常中 e.getMessage() 和 e.toString() e.printStackTrace()的区别常见的几种异常

Java 异常中 e.getMessage() 和 e.toString() e.printStackTrace()的区别 一、概述 在java异常体系中&#xff0c;要打印异常信息&#xff0c;可以通过&#xff1a;e.getMessage() 、 e.toString() e.printStackTrace() 等方法打印出 一些 异常信息。已知的是这些方法都可以打…

WinBUGS对多元随机波动率模型:贝叶斯估计与模型比较

在本文中&#xff0c;我们通过一个名为WinBUGS的免费贝叶斯软件&#xff0c;可以很容易地完成基于似然的多变量随机波动率&#xff08;SV&#xff09;模型的估计和比较。 最近我们被客户要求撰写关于随机波动率的研究报告&#xff0c;包括一些图形和统计输出。通过拟合每周汇率…

机器学习笔记之贝叶斯线性回归(一)线性回归背景介绍

机器学习笔记之贝叶斯线性回归——线性回归背景介绍引言回顾&#xff1a;线性回归场景构建从概率密度函数认识最小二乘法回顾&#xff1a;最小二乘估计回顾&#xff1a;线性回归与正则化关于线性回归的简单小结贝叶斯线性回归贝叶斯方法贝叶斯方法在线性回归中的任务贝叶斯线性…

kubernetes深入理解Pod对象之调度篇

目录 一、Pod调度流程 二、 容器资源限制 2.1 内存和CPU限制 三、 NodeSelector 四、NodeAffinity 4.1 基本概念 4.2 Pod 示例 4.2.1使用首选的节点亲和性调度 Pod 4.2.2依据强制的节点亲和性调度 Pod 五、Taints与Tolerations 5.1 基本概念 5.2Taints与Toleratio…

Ceph块存储

目录 一、环境准备 二、什么是块存储 三、创建块共享 1、查看存储池 2、创建镜像、查看镜像 3、镜像扩容、缩容 四、客户端通过KRBD访问共享镜像 1、客户端安装 2、客户端配置 3、客户端获取镜像 4、客户端写入数据 五、快照 1、查看、创建快照 2、还原快照 六、…