文件描述符
Linux系统中一切皆文件,系统中一切都被抽象成文件。对这些文件的读写都需要通过文件描述符来完成。这些文件描述符存储在文件描述符表中,内核为每一个进程维护了一个文件描述符表。文件描述符表能够存储的打开的文件数是有限制的,默认为1024个。
当一个进程被启动之后,内核PCB的文件描述符表中就已经分配了三个文件描述符:
STDIN_FILENO
:标准输入,通过该文件描述符将数据输入到终端文件。STDOUT_FILENO
:标准输出,通过该文件描述符将数据通过终端输出。STDERR_FILENO
:标准错误,通过该文件描述符将错误信息通过终端输出。
管道
管道的本质是内核中的一块内存(内核缓冲区),管道数据通过循环队列来维护,管道对应的内核缓冲区大小为64KB。
- 管道分为读端和写端,管道中的数据只能从写端流向读端,要实现双向同时通信需要设置两个管道。
- 管道中的数据只能读一次,数据一旦被读出就从管道中被抛弃。
- 对管道的读写操作默认是阻塞的。
- 管道内部保证同步机制。
- 数据以字节流的形式写入管道。
读管道,需要根据写端的状态进行分析。
- 写端未关闭
A. 管道无数据 -> 阻塞。如果管道中被写入了数据,阻塞解除,继续读。
B. 管道有数据 -> 不阻塞。如果管道中的数据被读完了,再继续读,管道会阻塞。 - 写端关闭
A.管道无数据 -> 阻塞解除,read函数返回0。
B.管道有数据 -> read函数先将数据读出,数据读完之后read函数返回0,阻塞解除。
写管道,需要根据读端的状态进行分析。
- 读端未关闭
A. 管道未写满 -> 不阻塞。一直写数据。
B. 管道写满 -> 阻塞。当读端将管道数据读走后,阻塞解除,继续写。 - 读端关闭
A.管道破裂,收到信号SIGPIPE
,进程异常终止。
匿名管道
匿名管道只能实现有血缘关系的进程间通信。
// 子进程写数据,父进程读数据
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
#include <string.h>
#include <stdlib.h>
int main()
{
// 1. 创建匿名管道
int fd[2];
int ret = pipe(fd);
if (ret == -1)
{
perror("pipe error");
exit(0);
}
// 2. 创建子进程 -> 父进程中能够操作管道的文件描述符也被复制到子进程中
pid_t pid = fork();
// 子进程
if (pid == 0)
{
// 关闭管道读端
close(fd[0]);
// 文件描述符的重定向,默认输出到终端,设置输出到管道
dup2(fd[1], STDOUT_FILENO);
// 3. 在子进程中执行 execlp("ps", "ps", "aux", NULL);
execlp("ps", "ps", "aux", NULL);
perror("execlp error");
}
// 父进程
else if (pid > 0)
{
// 关闭管道写端
close(fd[1]);
// 4.父进程读管道