Linux C语言 39-进程间通信IPC之管道

Linux C语言 39-进程间通信IPC之管道

本节关键字:C语言 进程间通信 管道 FIFO
相关库函数:pipe、mkfifo、mknod、write、read

什么是管道?

管道通常指“无名管道”,是Unix系统中最古老的IPC通信方式。

管道的分类

  • 管道(无名管道)
  • FIFO(命名管道)

管道的特点

管道(无名管道)
  • 半双工,数据只能由连接的一端发送到另一端,并且读端和写端固定;
  • 只能用于具有亲缘关系的进程通信,如父子进程、兄弟进程;
  • 可以当成一种特殊的文件,不属于任何文件系统,只存在于内存中。读写使用read()和write等函数;
  • 管道有互斥功能,同一时间只能有一个进程从管道读取数据。
FIFO(命名管道)
  • FIFO是一种文件类型,有路径名,以一种特殊设备文件形式存在于文件系统中;
  • FIFO可以在无关的进程之间交换数据,这点与无名管道不同;
  • 管道有互斥功能,同一时间只能有一个进程从管道读取数据。

管道相关库函数

管道(无名管道)
#include <unistd.h>
int pipe(int fd[2]);
/**
@brief 创建管道,需要手动关闭管道,即手动关闭两个文件描述符
@param fd 承接管道创建时的两个文件描述符,fd[0]为读端描述符,fd[1]为写端描述符
@return 成功返回0,失败返回-1并设置错误码error

错误码error类型:
EFAULT     pipefd无效
EINVAL     (pipe2())标志中的值无效
EMFILE  进程正在使用的文件描述符过多
ENFILE     打开文件总数已达到系统限制
*/
管道使用例程
// 管道使用例程:创建管道和进程,子进程读管道,父进程写管道
#include <sys/wait.h>
#include <assert.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>

int main(int argc, char *argv[])
{
    int pipefd[2];
    pid_t cpid;
    char buf;

    assert(argc == 2);    // 断言:有一个命令行参数,argv[1]将被写入管道

    if (pipe(pipefd) == -1) 
    {
        perror("pipe");
        exit(EXIT_FAILURE);
    }

    cpid = fork();
    if (cpid == -1) 
    {
        perror("fork");
        exit(EXIT_FAILURE);
    }

    if (cpid == 0)        // 子进程:关闭写文件描述符,读管道
    {
        close(pipefd[1]);

        while (read(pipefd[0], &buf, 1) > 0)
            write(STDOUT_FILENO, &buf, 1);

        write(STDOUT_FILENO, "\n", 1);
        close(pipefd[0]);
        _exit(EXIT_SUCCESS);

    } 
    else                   // 父进程:关闭读文件描述符,写管道,将命令行参数argv[1]写入管道
    {
        close(pipefd[0]);
        write(pipefd[1], argv[1], strlen(argv[1]));
       
        close(pipefd[1]);  // 读端将收到EOF
        wait(NULL);        // 等待子进程退出
        exit(EXIT_SUCCESS);
    }
   
    return 0;
}
FIFO(命名管道)
#include <sys/stat.h>
int mkfifo(const char *pathname, mode_t mode);
/**
@brief 创建FIFO管道,需要手动关闭管道,即手动关闭文件描述符。FIFO特殊文件不是匿名通信通道,而是通过调用mkfifo()输入到文件系统中
@param pathname FIFO的路径名,文件真实存在
@param mode 创建FIFO文件的权限,创建的文件的权限为(mode&~umask)
@return 成功返回0,失败返回-1并设置错误码error

错误码error分类:
EACCES        路径名中的一个目录没有搜索(执行)权限
EEXIST        路径名已存在。这包括路径名是符号链接的情况,无论是否悬空
ENAMETOOLONG    路径名的总长度大于PATH_MAX,或者单个文件名组件的长度大于NAME_MAX。在GNU系统中,对文件名的总长度没有强制限制,但一些文件系统可能会对组件的长度进行限制
ENOENT        路径名中的目录组件不存在,或者是悬挂的符号链接
ENOSPC        目录或文件系统没有空间容纳新文件
ENOTDIR        在路径名中用作目录的组件实际上不是目录
EROFS        路径名是指只读文件系统

Trip:一旦你以这种方式创建了一个FIFO特殊文件,任何进程都可以像普通文件一样打开它进行读取或写入。然而,在您可以继续对其进行任何输入或输出操作之前,它必须同时在两端打开。打开FIFO以正常读取块,直到其他进程打开相同的FIFO进行写入,反之亦然。有关fifo特殊文件的非阻塞处理,请参见fifo(7)。
*/
FIFO使用例程

FIFO使用例程:创建FIFO,多个个服务端读,多个个客户端写,验证管道的互斥功能

FIFO使用服务端例程
// FIFO使用例程 服务端,循环读取管道中的信息,直到管道的所有写端关闭
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/types.h>

int main(void)
{
    char pathname[] = "./myfifo";
    int ret, fd;
    char buffer[1024];
    
    umask(0);
    ret = mkfifo(pathname, 0666);  // mode附加读写权限
    fd = open(pathname, O_RDONLY); // 服务端只读打开FIFO文件
    if (fd < 0)
        return -1;
    
    for ( ; ; )
    {
        bzero(buffer, 0);
        ret = read(fd, buffer, sizeof(buffer)-1);
        if (ret > 0)
        {
            printf("server recv: %s\n", buffer);
            fflush(stdout);
        }
        else if (ret == 0)
        {
            printf("server: client quit\n");
            break;
        }
        else
        {
            perror("server failed to read myfifo\n");
            break;
        }
    }
    
    close(fd);
    
    return 0;
}
FIFO使用客户端例程
// FIFO使用例程:客户端,将用户输入的信息写入管道,循环执行6次退出
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/types.h>

int main(void)
{
    char pathname[] = "./myfifo";
    int ret, fd, cnt;
    char buffer[1024];
    
    fd = open(pathname, O_WRONLY);    // 客户端只写开打FIFO文件
    if (fd < 0)
        return -1;
    
    do
    {
        printf("please input message: ");
        fflush(stdout);
        
        bzero(buffer, 0);
        ret = read(STDIN_FILENO, buffer, sizeof(buffer)-1);
        if (ret <= 0)
        {
            perror("read");
        }
        else
        {
            write(fd, buffer, ret);
        }
        
    } while (++cnt <= 5);
    
    close(fd);
    return 0;
}
运行结果
/// 从两个服务端的运行结果可知,管道有互斥功能,同一时刻只有一个进程可以从管道读取数据

// 服务端1:
$ ./fifoserver 
server recv: client1: 1

server recv: client2: 1

server recv: client1: 2

server recv: client1: 3

server recv: client1: 5

server recv: client2: 4

server recv: client2: 6

server: client quit

// 服务端2:
$ ./fifoserver 
server recv: client1: 4

server recv: client1: 6

server recv: client2: 2

server recv: client2: 3

server recv: client2: 5

server: client quit

// 客户端1:
$ ./fifoclient 
please input message: client1: 1    
please input message: client1: 2
please input message: client1: 3
please input message: client1: 4
please input message: client1: 5
please input message: client1: 6

// 客户端2:
$ ./fifoclient 
please input message: client2: 1
please input message: client2: 2     
please input message: client2: 3
please input message: client2: 4
please input message: client2: 5
please input message: client2: 6

启动server后使用ll命令查看目录文件信息,可以看到已经创建了myfifo文件
FIFO的创建

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

L1-027:出租

题目描述 下面是新浪微博上曾经很火的一张图&#xff1a; 一时间网上一片求救声&#xff0c;急问这个怎么破。其实这段代码很简单&#xff0c;index数组就是arr数组的下标&#xff0c;index[0]2 对应 arr[2]1&#xff0c;index[1]0 对应 arr[0]8&#xff0c;index[2]3 对应 arr…

绝地求生:【PC】12月网页活动和特殊空投

亲爱的玩家朋友们&#xff0c;大家好&#xff01; 将为2023年画上圆满句号的大型活动和丰厚奖励已经准备就绪。参与活动即可获得武器皮肤&#xff0c;服装&#xff0c;喷漆&#xff0c;黑货票券在内的奖品&#xff0c;赶快来参与活动获得奖励吧&#xff01; 荣都上线纪念活动 …

JavaSE基础50题:19. 递归求斐波那契数列的第N项。

概述 用递归求斐波那契数列的第N项。 斐波那契数列&#xff1a; 1 1 2 3 5 8 …… f(n) f(n-1) f(n-2) 代码 public class P19 {public static int fibnacio(int n) {if (n 1 || n 2) {return 1;}int tmp fibnacio(n-1) fibnacio(n-2);return tmp;}public static void…

LLM之Agent(三):HuggingGPT根据用户需求自动调用Huggingface合适的模型

​ 浙大和微软亚洲研究院开源的HuggingGPT&#xff0c;又名JARVIS&#xff0c;它可以根据用户的自然语言描述的需求就可以自动分析需要哪些AI模型&#xff0c;然后去Huggingface上直接调用对应的模型&#xff0c;最终给出用户的解决方案。 一、HuggingGPT的工作流程 它的…

六个自媒体写作方法,提升自媒体创作收益

在自媒体时代&#xff0c;写作成为了一个不可或缺的技能。特别是对于新手来说&#xff0c;掌握一些有效的写作方法&#xff0c;可以事半功倍&#xff0c;更好地展现个人创意和观点。在这里&#xff0c;我将分享六个适合新手的自媒体写作方法&#xff0c;希望能够为你在写作之路…

外汇天眼:什么时段做外汇交易最好,有所谓的“最佳时间点”吗?

在外汇交易的时候&#xff0c;很多手动交易的投资者不知道到底什么时间段操作交易最适合自己。 我们在进行选择最佳交易时间的时候&#xff0c;一定要明白各时间段的全球各个市场的交易状况&#xff0c;这样你才能分配好自己的时间。 当然在通过技术分析与基本分析选择好币种后…

刚刚,深兰科技在国际顶级会议EMNLP竞赛中再夺两冠

12月6日至10日&#xff0c;自然语言处理领域的国际顶级会议EMNLP(自然语言处理中的经验方法会议)在新加坡召开&#xff0c;研究人员、学者和业界专业人士齐聚一堂&#xff0c;展示和讨论该领域的最新研究成果、进展和创新。 会上公布了各项任务竞赛的获奖名单&#xff0c;深兰科…

十分钟带你搞懂Python那啥

Python爬虫的用处就不需要我多说了吧&#xff0c;今天就来带大家十分钟快速学会Python是如何爬取网页信息的&#xff0c;当然大家在爬取目标网页内容之前一定要遵守该网页的爬虫规则&#xff0c;以免带来不必要的麻烦&#xff0c;因而本次的示例所爬取的网页也是自己的本地网站…

【Vulnhub 靶场】【Funbox: GaoKao】【简单】【20210606】

1、环境介绍 靶场介绍&#xff1a;https://www.vulnhub.com/entry/funbox-gaokao,707/ 靶场下载&#xff1a;https://download.vulnhub.com/funbox/FunboxGaoKao.ova 靶场难度&#xff1a;简单 发布日期&#xff1a;2021年06月06日 文件大小&#xff1a;1.3 GB 靶场作者&#…

专业130+总分400+云南大学通信847专业基础综考研经验(原专业课827)

今年专业130总分400云南大学通信上岸&#xff0c;整体考研感觉还是比较满意&#xff0c;期间也付出了很多心血&#xff0c;走过弯路&#xff0c;下面分享一下这一年考研得失&#xff0c;希望大家可以从中有所借鉴。 先说明我在考研报名前更换成云南大学的理由&#xff1a;&…

通过项目管理软件监管新员工入职流程的方法与策略

项目管理软件是什么&#xff1f;项目管理软件都能做什么&#xff1f;是不是只有项目团队需要啊&#xff1f;NO&#xff01;项目管理软件乍听其名不免让人觉得这不过是个项目领域的专用工具。 那什么是项目呢&#xff1f;项目是为创造独特的产品、服务或成果而进行的体系化的工…

Qt内存管理、UI编辑器、客制化组件、弹出对话框、常用部件类

头文件的小技巧 #include <QtWidgets> // 在自动生成的 .h 里面加上此句 适用条件&#xff1a; QT 的内存管理 当父窗体被关闭时&#xff0c;子部件的内存会自动释放。 对象树是一种管理对象生命周期的机制。当一个对象被添加到另一个对象的子对象列表中时&#xff0…

机器学习应用 | 使用 MATLAB 进行异常检测(下)

在使用MATLAB 进行异常检测&#xff08;上&#xff09;中&#xff0c;我们探讨了什么是异常值&#xff0c;简单的一维数据异常检测问题&#xff0c;针对高维数据的有监督异常检测方法。 在&#xff08;下&#xff09;篇中&#xff0c;我们将和大家一起探讨无监督异常检测。 没…

“身份证信息批量核验:高效解决管理难题,轻松提升工作效率“

尊敬的读者们&#xff0c;您是否曾经因为身份证信息的核验而感到烦恼&#xff1f;是否曾经因为手动核验而感到繁琐和耗时&#xff1f;现在&#xff0c;我们向您介绍一款全新的工具——身份证信息批量核验器&#xff0c;它将帮助您一键解决管理难题&#xff0c;让工作事半功倍&a…

netty源码:(9)ChannelId

ChannelId代表了一个Channel的全局标识符。它提供了asShortText和asLongText两个方法来返回对应的字符串

C语言printf的输出格式大全及颜色字体打印

不同类型的数据有不同形式的打印格式&#xff0c;熟知了printf的不同输出格式会让后边的学习事半功倍。 文章目录 %d%i打印十进制整形数据标志&#xff08;flags&#xff09;%u打印无符号十进制%o 打印无符号八进制%x %X打印大写或小写十六进制%f打印浮点数%e %E科学计数法打印…

Java几种深拷贝方式比较

Java的深度拷贝分为克隆&#xff08;实现Java的Clone接口&#xff09;和序列化&#xff08;实现Java的Serializable接口&#xff09;两种&#xff0c;由于序列化有不同的方式&#xff0c;下面分析一下每种的注意事项和性能对比。 一、继承Cloneable接口 可以使用Java提供的Clo…

Java基础50题: 21.实现一个方法printArray, 以数组为参数,循环访问数组中的每个元素,打印每个元素的值.

概述 实现一个方法printArray, 以数组为参数,循环访问数组中的每个元素,打印每个元素的值. 代码 public static void printArray(int[] array) {for (int i 0; i < array.length; i) {System.out.println(array[i] " ");}System.out.println();}public static…

class060 拓扑排序的扩展技巧【算法】

class060 拓扑排序的扩展技巧【算法】 算法讲解060【必备】拓扑排序的扩展技巧 2023-12-7 22:23:02 code1 P4017 最大食物链计数 // 最大食物链计数 // a -> b&#xff0c;代表a在食物链中被b捕食 // 给定一个有向无环图&#xff0c;返回 // 这个图中从最初级动物到最顶…

Atlassian Confluence 模板注入代码执行漏洞风险通告

近期&#xff0c;亚信安全CERT通过监控发现&#xff0c;Atlassian 公司发布了一则安全公告&#xff0c;针对 Confluence 数据中心和 Confluence 服务器存在的远程代码执行漏洞&#xff08;CVE-2023-22522&#xff09;进行了修复。该漏洞涉及 Confluence 页面中的模板注入问题&a…