▋****1. 管道的概念

管道,又名「无名管理」,或「匿名管道」,管道是一种非常基本,也是使用非常频繁的IPC方式。

1.1 管道本质

  • 管道的本质也是一种文件,不过是伪文件,实际上是一块内核缓冲区,大小4K;
  • 管道创建以后会产生两个文件描述符,一个是读端,另一个是写端;
  • 管道里的数据只能从写端被写入,从读端被读出;

1.2 管道原理

管道是内核的一块缓冲区,更具体一些,是一个环形队列。数据从队列的一端写入数据,另一端读出,如下图示:

1.3 管道的优点

简单

1.4 管道的缺点

  • 只能单向通信,如果需要双向通信则需要建立两个管道;
  • 只能应用于具有血缘关系的进程,如父子进程;
  • 缓冲区大小受限,通常为1页,即4k;

▋****2. 管道的创建

管道创建三步曲:

a. 父进程调用pipe函数创建管道;

b. 父进程调用fork函数创建子进程;

c. 父进程关闭fd[0],子进程关闭fd[1];

具体如下图所示:

▋****3. 管道的读写行为

a. 管道的缓冲区大小固定为4k,所以如果管道内数据已经写满,则无法再写入数据,进程的write调用将阻塞,直到有足够的空间再写入数据;

b. 管道的读动作比写动作要快,数据一旦被读走了,管道将释放相应的空间,以便后续数据的写入。当所有的数据都读完之后,进程的read()调用将阻塞,直到有数据再次写入。

▋****4. 例程

父子间通信:

 1#include <stdio.h>
2#include <sys/types.h>
3#include <unistd.h>
4#include <string.h>
5
6int main()
7{
8 int fd[2];
9 pid_t pid;
10 char buf[1024];
11 char *data = "hello world!";
12
13 /* 创建管道 */
14 if (pipe(fd) == -1) {
15 printf("ERROR: pipe create failed!\n");
16 return -1;
17 }
18
19 pid = fork();
20 if (pid == 0) {
21 /* 子进程 */
22 close(fd[1]); // 子进程读取数据,关闭写端
23 read(fd[0], buf, sizeof(buf)); // 从管道读数据
24 printf("child process read: %s\n", buf);
25 close(fd[0]);
26 } else if (pid > 0) {
27 /* 父进程 */
28 close(fd[0]); //父进程写数据,关闭读端
29 write(fd[1], data, strlen(data)); // 向管道写数据
30 printf("parent process write: %s\n", data);
31 close(fd[1]);
32 }
33
34 return 0;
35}

兄弟间通信:

 1#include <stdio.h>
2#include <sys/types.h>
3#include <unistd.h>
4#include <string.h>
5#include <sys/wait.h>
6
7int main ()
8{
9 int fd[2];
10 int i = 0;
11 pid_t pid;
12 char buf[1024];
13 char *data = "hello world!";
14
15 /* 创建管道 */
16 if (pipe(fd) == -1) {
17 printf("ERROR: pipe create failed!\n");
18 return -1;
19 }
20
21 for (i = 0; i < 2; i++) {
22 pid = fork();
23 if (pid == -1) {
24 printf("ERROR: fork error!\n");
25 return -1;
26 } else if (pid == 0) {
27 break;
28 }
29 }
30
31 /* 通过i来判断创建的子进程及父进程 */
32 if (i == 0) {
33 /* 第一个子进程,兄进程 */
34 close(fd[0]); // 兄进程向弟进程写数据,关闭读端
35 write(fd[1], data, strlen(data));
36 printf("elder brother send: %s\n", data);
37 close(fd[1]);
38 } else if (i == 1) {
39 /* 第二个子进程,弟进程 */
40 close(fd[1]);
41 read(fd[0], buf, sizeof(buf));
42 printf("younger brother receive: %s\n", buf);
43 close(fd[0]);
44 } else {
45 /* 父进程 */
46 close(fd[0]);
47 close(fd[1]);
48 for (i = 0; i < 2; i++) {
49 wait(NULL);
50 }
51 }
52
53 return 0;
54}

更多精彩内容,请关注公众号良许Linux,公众内回复1024可免费获得5T技术资料,包括:Linux,C/C++,Python,树莓派,嵌入式,Java,人工智能,等等。公众号内回复进群,邀请您进高手如云技术交流群。


公众号:良许Linux

有收获?希望老铁们来个三连击,给更多的人看到这篇文章

Linux系统编程—有名管道的更多相关文章

  1. linux系统编程之管道(三)

    今天继续研究管道的内容,这次主要是研究一下命名管道,以及与之前学过的匿名管道的区别,话不多说,进入正题: 所以说,我们要知道命名管道的作用,可以进行毫无关系的两个进程间进行通讯,这是匿名管道所无法实现 ...

  2. linux系统编程之管道(三):命令管道(FIFO)

    一,匿名管道PIPE局限性 管道的主要局限性正体现在它的特点上: 只支持单向数据流: 只能用于具有亲缘关系的进程之间: 没有名字: 管道的缓冲区是有限的(管道制存在于内存中,在管道创建时,为缓冲区分配 ...

  3. linux系统编程之管道(二):管道读写规则

    一,管道读写规则 当没有数据可读时 O_NONBLOCK disable:read调用阻塞,即进程暂停执行,一直等到有数据来到为止. O_NONBLOCK enable:read调用返回-1,errn ...

  4. linux系统编程之管道(一):匿名管道(pipe)

    一,什么是管道 管道是Linux支持的最初Unix IPC形式之一,具有以下特点: 管道是半双工的,数据只能向一个方向流动:需要双方通信时,需要建立起两个管道: 只能用于父子进程或者兄弟进程之间(具有 ...

  5. linux系统编程之管道(二)

    今天继续研究管道,话不多说,言归正传: 对于管道,有一定的读写规则,所以这里主要是对它的规则进行探讨,具体规则如下: 规则一: 下面用程序来验证下,还是用上节学的子进程写数据,父进程读取数据的例子,只 ...

  6. Linux 系统编程 学习:02-进程间通信1:Unix IPC(1)管道

    Linux 系统编程 学习:02-进程间通信1:Unix IPC(1)管道 背景 上一讲我们介绍了创建子进程的方式.我们都知道,创建子进程是为了与父进程协作(或者是为了执行新的程序,参考 Linux ...

  7. Linux系统编程@进程通信(一)

    进程间通信概述 需要进程通信的原因: 数据传输 资源共享 通知事件 进程控制 Linux进程间通信(IPC)发展由来 Unix进程间通信 基于System V进程间通信(System V:UNIX系统 ...

  8. linux系统编程之框架

    linux系统编程之框架: 1. 进程 1.1 进程概念 1.1.1 PCB 1.1.2 环境变量 1.2 进程控制 1.3 进程间通信 1.3.1 管道 1.3.2 有名管道 1.3.3 共享内存 ...

  9. Linux 系统编程 学习:05-进程间通信2:System V IPC(2)

    Linux 系统编程 学习:05-进程间通信2:System V IPC(2) 背景 上一讲 进程间通信:System V IPC(1)中,我们介绍了System IPC中有关消息队列.共享内存的概念 ...

随机推荐

  1. P4719 【模板】"动态 DP"&动态树分治

    题目描述 给定一棵 n 个点的树,点带点权. 有 m 次操作,每次操作给定 x,y,表示修改点 x 的权值为 y. 你需要在每次操作之后求出这棵树的最大权独立集的权值大小. 输入格式 第一行有两个整数 ...

  2. c# 可获取系统环境

    c# 可获取系统环境, 启动进程执等 *.shell [MenuItem("Tools/DesignHelper/Clean and Pull")]     private sta ...

  3. 浅析一个lua文件窥slua工作机制

    slua的东西不是几句话能讲得完,这里只说结论不说原因,原因有空写个Little Slua工程来解释,下面注释中有几个关键点:LuaVar系列类:LuaFunction,LuaTable,LuaDel ...

  4. webpack3.10.0(入门系列基本概念1)

    一.概念 webpack的核心是一个用于现代JavaScript应用程序的静态模块打包程序.当webpack处理您的应用程序时,它会递归地构建一个依赖图,其中包含应用程序所需的每个模块,然后将所有这些 ...

  5. Activiti7 查询用户任务列表

    package com.itheima.activiti; import org.activiti.engine.ProcessEngine; import org.activiti.engine.P ...

  6. [业界方案] 用SOFATracer学习分布式追踪系统Opentracing

    [业界方案] 用SOFATracer学习分布式追踪系统Opentracing 目录 [业界方案] 用SOFATracer学习分布式追踪系统Opentracing 0x00 摘要 0x01 缘由 &am ...

  7. SqlAnalyzer1.00源码

    SQL解析的夙愿今天终于完成,但限于SQL远超算术表达式,Json,XML等的复杂度,只是解析了一部分.形成普适性的SQL解析,仍需倾注精力. 代码下载:https://files.cnblogs.c ...

  8. 基于STM32的脉搏心率检测仪(OLED可以实时显示脉冲波形)

    —设计完整,功能可全部实现,有完整报告文档说明.程序以及pcb文件— 可作为:课程设计,STM32实践学习,电子制作等 设计所实现的功能: 利用STM32的AD采集功能实时采集心率传感器信号输出引脚输 ...

  9. LC算法技巧总结(二):双指针和滑动窗口技巧

    我把双指针技巧再分为两类,一类是「快慢指针」,一类是「左右指针」.前者解决主要解决链表中的问题,比如典型的判定链表中是否包含环:后者主要解决数组(或者字符串)中的问题,比如二分查找. 一.快慢指针的常 ...

  10. Oracle数据库之表与表数据操作

    一.SQL语言 SQL语言分为四种,分别是:数据定义语言(DDL).数据操纵语言(DCL).事务控制语言(TCL).数据控制语言(DML). 1.1 数据定义语言(DDL) 建立.修改.删除数据库对象 ...