Linux:命名管道(进程间通信六)

我们学习了匿名管道,我们知道匿名管道只能适用于血缘关系间的通信(比如父子),但是我们想要在两个毫不相关的进程间通信,又该怎么办呢??于是我们的命名管道,开始表演他的独门绝技--打开同一路径下的同名文件(命名管道文件) 实现通信

1.命名管道初始

我先提一个问题,进程 A 和 B 都打开 a/b/c.txt,内核中,操作系统会不会加载两次a/b/c.txt,答案是不会

因为这也太浪费资源了,我们前面学过,假如A打开了一个.txt,那么就会创建出对应的struct_file,此时struct_file指向打开的文件,如果我们再用进程B打开这个文件,虽然B也会创建一个struct_file,但是也是指向打卡文件

也就是说,A,B指向的东西是一样的,这部就满足了让两个进程看到同一份内存资源这个通信前提吗!!但是我们只想要打开该文件进行写入读取,实现通信,并不想将这个文件刷新回磁盘,所以我们有了管道文件fifo

所以我们知道,FIFO 管道文件:本质是 "伪文件"------ 它在文件系统中只有路径(用于进程定位),但所有读写操作仅在 "内核缓冲区" 中完成,数据永不刷盘

为了区别匿名管道,我们将这种管道叫做命名管道

2.用命令行mkfifo创建命名管道并进行通信

linux命令行支持用mkfifo直接创建命名管道文件

1.使用mkfifo [NAME]创建一个fifo文件,可以发现开头是p,就说明创建的是管道文件

2.使用echo来表示一个进程,linux下一切操作都是进程,echo可以理解为进程A向管道进行写入操作

可是你会发现,为什么堵塞了呢?那是因为命名管道(FIFO)的通信需要读端和写端同时存在------ 默认情况下,FIFO 的读写操作是 "阻塞模式":

  • 当你执行 echo "你好" > fifo 时,这是一个写操作,会尝试向 FIFO 写入数据
  • 但此时没有任何进程打开这个fifo作为读端 ,写操作会被内核阻塞,直到有另一个进程以 "读模式" 打开fifo,写操作才能完成

3.打开另一个终端,并且进入当前目录下,执行cat < fifo操作,模拟进程B读取fifo数据

成功进行了两个不同进程间的通信

3.使用系统函数实现两个进程通信

server.cc:

cpp 复制代码
#include <iostream>
#include <string>
#include <sys/stat.h>
#include <sys/types.h>
#include "common.hpp"
#include <fcntl.h>
#include <unistd.h>

int main()
{
    umask(0);
    int n = mkfifo(FIFO_FILE, 0666);
    if (n != 0)
    {
        std::cerr << "mkfifo error" << std::endl;
        return 1;
    }
    int fd = open(FIFO_FILE, O_RDONLY);
    if (fd == -1)
    {
        std::cerr << "open fifo error" << std::endl;
        return 2;
    }
    char buffer[1024];
    while (true)
    {
        n = read(fd, buffer, sizeof(buffer) - 1);
        if (n > 0)
        {
            buffer[n] = '\0';
            std::cout << "client say : " << buffer << std::endl;
        }
    }
    close(fd);
    return 0;
}

client.cc

cpp 复制代码
#include <iostream>
#include <string>
#include <sys/stat.h>
#include <sys/types.h>
#include "common.hpp"
#include <fcntl.h>
#include <unistd.h>

int main()
{
    int fd = open(FIFO_FILE, O_WRONLY);
    if (fd == -1)
    {
        std::cerr << "open fifo error" << std::endl;
        return 2;
    }
    std::string buffer;
    while (true)
    {
        std::cout << "client : ";
        std::getline(std::cin, buffer);
        write(fd, buffer.c_str(), sizeof(buffer));
    }
    close(fd);
    return 0;
}

common.cc

cpp 复制代码
#pragma once
#define FIFO_FILE "fifo"

Makefile:

cpp 复制代码
.PHONEY : all
all : client server
client : client.cc
	g++ client.cc -o client
server : server.cc
	g++ server.cc -o server

.PHONEY : clean
clean : 
	rm -f client server fifo

我们需要先使用./server来创建管道,不然直接使用./client会因为没有创建管道而报错哦

下面是运行结果:

成功实现两个不相干进程中的通信

这就是命名管道的基本知识啦,下篇博客我们讲深入理解命名管道哦,敬请期待~~

相关推荐
在路上看风景2 小时前
19. 成员初始化列表和初始化对象
c++
zmzb01032 小时前
C++课后习题训练记录Day98
开发语言·c++
wdfk_prog2 小时前
[Linux]学习笔记系列 -- [drivers][input]input
linux·笔记·学习
念风零壹2 小时前
C++ 内存避坑指南:如何用移动语义和智能指针解决“深拷贝”与“内存泄漏”
c++
七夜zippoe2 小时前
CANN Runtime任务描述序列化与持久化源码深度解码
大数据·运维·服务器·cann
盟接之桥2 小时前
盟接之桥说制造:引流品 × 利润品,全球电商平台高效产品组合策略(供讨论)
大数据·linux·服务器·网络·人工智能·制造
忆~遂愿3 小时前
ops-cv 算子库深度解析:面向视觉任务的硬件优化与数据布局(NCHW/NHWC)策略
java·大数据·linux·人工智能
湘-枫叶情缘3 小时前
1990:种下那棵不落叶的树-第6集 圆明园的对话
linux·系统架构
孞㐑¥3 小时前
算法——BFS
开发语言·c++·经验分享·笔记·算法
Fcy6484 小时前
Linux下 进程(一)(冯诺依曼体系、操作系统、进程基本概念与基本操作)
linux·运维·服务器·进程