# 管道,第 1 部分:管道介绍
> 原文:<https://github.com/angrave/SystemProgramming/wiki/Pipes%2C-Part-1%3A-Introduction-to-pipes>
## 什么是 IPC?
进程间通信是一个进程与另一个进程通信的任何方式。你已经看过这种虚拟内存的一种形式了!一段虚拟内存可以在父母和孩子之间共享,从而实现通信。您可能希望将该内存包装在`pthread_mutexattr_setpshared(&attrmutex, PTHREAD_PROCESS_SHARED);`互斥锁(或进程宽互斥锁)中以防止竞争条件。
有更多标准的 IPC 方式,比如管道!考虑您是否在终端中键入以下内容
```source-shell
$ ls -1 | cut -d'.' -f1 | uniq | sort | tee dir_contents
```
以下代码的作用是什么(如果你愿意,你可以跳过这个并不重要)?那么它`ls`是当前目录(-1 表示它每行输出一个条目)。然后`cut`命令在第一个周期之前获取所有内容。 Uniq 确保所有行都是 uniq,排序将它们和 tee 输出排序到文件。
重要的是,bash 创建 **5 个独立的进程**,并将它们的标准出口/标准连接到管道,其中的跟踪看起来像这样。
(0)ls(1)------&gt;(0)cut(1)-------&gt;(0)uniq(1)------&gt;(0)排序(1)------&gt;(0)tee(1)
管道中的数字是每个进程的文件描述符,箭头表示重定向或管道输出的位置。
## 什么是管道?
POSIX 管道几乎就像它的真正对应物 - 你可以在一端填充字节,它们将以相同的顺序出现在另一端。然而,与真实管道不同,进程始终在同一方向,一个文件描述符用于读取,另一个文件描述符用于写入。 `pipe`系统调用用于创建管道。
```c
int filedes[2];
pipe (filedes);
printf("read from %d, write to %d\n", filedes[0], filedes[1]);
```
这些文件描述符可以与`read`一起使用 -
```c
// To read...
char buffer[80];
int bytesread = read(filedes[0], buffer, sizeof(buffer));
```
和`write` -
```c
write(filedes[1], "Go!", 4);
```
## 如何使用管道与子进程通信?
使用管道的常用方法是在分叉之前创建管道。
```c
int filedes[2];
pipe (filedes);
pid_t child = fork();
if (child > 0) { /* I must be the parent */
char buffer[80];
int bytesread = read(filedes[0], buffer, sizeof(buffer));
// do something with the bytes read
}
```
然后,孩子可以将消息发送回父母:
```c
if (child == 0) {
write(filedes[1], "done", 4);
}
```
## 我可以在一个过程中使用管道吗?
简短回答:是的,但我不确定你为什么要大声笑!
这是一个向自己发送消息的示例程序:
```c
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
int main() {
int fh[2];
pipe(fh);
FILE *reader = fdopen(fh[0], "r");
FILE *writer = fdopen(fh[1], "w");
// Hurrah now I can use printf rather than using low-level read() write()
printf("Writing...\n");
fprintf(writer,"%d %d %d\n", 10, 20, 30);
fflush(writer);
printf("Reading...\n");
int results[3];
int ok = fscanf(reader,"%d %d %d", results, results + 1, results + 2);
printf("%d values parsed: %d %d %d\n", ok, results[0], results[1], results[2]);
return 0;
}
```
以这种方式使用管道的问题是写入管道可能阻塞,即管道仅具有有限的缓冲容量。如果管道已满,写入过程将阻止!缓冲区的最大大小取决于系统;典型值从 4KB 到 128KB。
```c
int main() {
int fh[2];
pipe(fh);
int b = 0;
#define MESG "..............................."
while(1) {
printf("%d\n",b);
write(fh[1], MESG, sizeof(MESG))
b+=sizeof(MESG);
}
return 0;
}
```
参见[管道,第 2 部分:管道编程秘密](/angrave/SystemProgramming/wiki/Pipes%2C-Part-2%3A-Pipe-programming-secrets)
- UIUC CS241 系统编程中文讲义
- 0. 简介
- #Informal 词汇表
- #Piazza:何时以及如何寻求帮助
- 编程技巧,第 1 部分
- 系统编程短篇小说和歌曲
- 1.学习 C
- C 编程,第 1 部分:简介
- C 编程,第 2 部分:文本输入和输出
- C 编程,第 3 部分:常见问题
- C 编程,第 4 部分:字符串和结构
- C 编程,第 5 部分:调试
- C 编程,复习题
- 2.进程
- 进程,第 1 部分:简介
- 分叉,第 1 部分:简介
- 分叉,第 2 部分:Fork,Exec,等等
- 进程控制,第 1 部分:使用信号等待宏
- 进程复习题
- 3.内存和分配器
- 内存,第 1 部分:堆内存简介
- 内存,第 2 部分:实现内存分配器
- 内存,第 3 部分:粉碎堆栈示例
- 内存复习题
- 4.介绍 Pthreads
- Pthreads,第 1 部分:简介
- Pthreads,第 2 部分:实践中的用法
- Pthreads,第 3 部分:并行问题(奖金)
- Pthread 复习题
- 5.同步
- 同步,第 1 部分:互斥锁
- 同步,第 2 部分:计算信号量
- 同步,第 3 部分:使用互斥锁和信号量
- 同步,第 4 部分:临界区问题
- 同步,第 5 部分:条件变量
- 同步,第 6 部分:实现障碍
- 同步,第 7 部分:读者编写器问题
- 同步,第 8 部分:环形缓冲区示例
- 同步复习题
- 6.死锁
- 死锁,第 1 部分:资源分配图
- 死锁,第 2 部分:死锁条件
- 死锁,第 3 部分:餐饮哲学家
- 死锁复习题
- 7.进程间通信&amp;调度
- 虚拟内存,第 1 部分:虚拟内存简介
- 管道,第 1 部分:管道介绍
- 管道,第 2 部分:管道编程秘密
- 文件,第 1 部分:使用文件
- 调度,第 1 部分:调度过程
- 调度,第 2 部分:调度过程:算法
- IPC 复习题
- 8.网络
- POSIX,第 1 部分:错误处理
- 网络,第 1 部分:简介
- 网络,第 2 部分:使用 getaddrinfo
- 网络,第 3 部分:构建一个简单的 TCP 客户端
- 网络,第 4 部分:构建一个简单的 TCP 服务器
- 网络,第 5 部分:关闭端口,重用端口和其他技巧
- 网络,第 6 部分:创建 UDP 服务器
- 网络,第 7 部分:非阻塞 I O,select()和 epoll
- RPC,第 1 部分:远程过程调用简介
- 网络复习题
- 9.文件系统
- 文件系统,第 1 部分:简介
- 文件系统,第 2 部分:文件是 inode(其他一切只是数据...)
- 文件系统,第 3 部分:权限
- 文件系统,第 4 部分:使用目录
- 文件系统,第 5 部分:虚拟文件系统
- 文件系统,第 6 部分:内存映射文件和共享内存
- 文件系统,第 7 部分:可扩展且可靠的文件系统
- 文件系统,第 8 部分:从 Android 设备中删除预装的恶意软件
- 文件系统,第 9 部分:磁盘块示例
- 文件系统复习题
- 10.信号
- 过程控制,第 1 部分:使用信号等待宏
- 信号,第 2 部分:待处理的信号和信号掩码
- 信号,第 3 部分:提高信号
- 信号,第 4 部分:信号
- 信号复习题
- 考试练习题
- 考试主题
- C 编程:复习题
- 多线程编程:复习题
- 同步概念:复习题
- 记忆:复习题
- 管道:复习题
- 文件系统:复习题
- 网络:复习题
- 信号:复习题
- 系统编程笑话