Linux学习笔记:
https://blog.csdn.net/2301_80220607/category_12805278.html?spm=1001.2014.3001.5482
前言:
在上一篇,我们已经讲解过文件描述符的相关问题了,但是今天,由于讲解重定向问题需要更进一步理解文件描述符的问题,所以我们先对文件描述符的问题进行一些补充讲解,然后再重点讲解一下重定向的问题。
目录
[1.1 文件描述符的类型](#1.1 文件描述符的类型)
[1.2 文件描述符的管理](#1.2 文件描述符的管理)
[1.3 文件描述符的相关操作](#1.3 文件描述符的相关操作)
[2.1 基本的重定向](#2.1 基本的重定向)
[2.2 重定向标准错误输出](#2.2 重定向标准错误输出)
[2.3 同时重定向标准输出和标准错误输出](#2.3 同时重定向标准输出和标准错误输出)
[2.4 输入输出重定向与管道](#2.4 输入输出重定向与管道)
[3.1 重定向的本质](#3.1 重定向的本质)
[3.2 dup2系统调用来实现重定向](#3.2 dup2系统调用来实现重定向)
[3.3 其它](#3.3 其它)
一、文件描述符
我们还是先再来讲解一遍文件描述符:在Linux中,每个打开的文件都会被分配一个文件描述符。文件描述符是一个非负整数,操作系统内核通过它来跟踪进程与文件之间的关联。文件描述符提供了与文件进行交互的通道,可以通过它执行各种文件操作。
1.1 文件描述符的类型
在Linux中,文件描述符被划分为三大类:
- 标准输入(stdin):文件描述符为0,表示程序的标准输入流,通常指向键盘输入。
- 标准输出(stdout):文件描述符为1,表示程序的标准输出流,通常指向终端。
- 标准错误(stderr):文件描述符为2,表示程序的标准错误流,也指向终端,通常用于输出错误信息。
除了这三个标准流,Linux允许通过系统调用打开其他文件或设备,从而获得新的文件描述符。这些文件描述符可以通过文件I/O操作进行读写。
1.2 文件描述符的管理
文件描述符是由操作系统内核管理的,它们与一个实际的文件表项(文件描述符表)相对应。每个进程都有一个与之关联的文件描述符表,用于记录当前进程打开的所有文件和设备的信息。
同时每个文件描述符也对应着一个描述该文件的内容及属性的结构体对象,这就是操作系统管理文件的本质
文件描述符 | 类型 | 描述 |
---|---|---|
0 | stdin | 标准输入流 |
1 | stdout | 标准输出流 |
2 | stderr | 标准错误流 |
3+ | 普通文件 | 可通过系统调用open() 获取的文件描述符 |
通过调用open()
系统调用,我们可以打开一个文件并获取其对应的文件描述符。open()
函数的基本原型如下:
cpp
int open(const char *pathname, int flags, mode_t mode);
pathname
:文件路径。flags
:打开文件时的标志,例如O_RDONLY
(只读)、O_WRONLY
(只写)、O_RDWR
(读写)等。mode
:文件权限,仅在创建文件时使用。
打开文件后,操作系统会返回一个文件描述符,用于后续的文件操作。
1.3 文件描述符的相关操作
在Linux中,文件描述符常用的操作包括:
- 读取文件 :通过
read()
系统调用读取文件内容。- 写入文件 :通过
write()
系统调用将数据写入文件。- 关闭文件 :通过
close()
系统调用关闭文件描述符。
这些系统调用接口我们在上一篇做过详细讲解了,这里就不再介绍了
二、输出重定向
输出重定向是将命令的标准输出(stdout)或者标准错误输出(stderr)重定向到文件、设备或其他进程的一个过程。Linux 提供了非常灵活的方式来实现输出重定向。首先我们先来看基本的重定向,也就是在命令行中可以直接实现的重定向
2.1 基本的重定向
在 Linux 中,我们可以使用 >
或 >>
来将输出重定向到文件。
>
:将输出重定向到文件,如果文件已存在则覆盖。>>
:将输出重定向到文件,如果文件已存在则追加。
例如,执行以下命令时:
cpp
echo "Hello, World!" > output.txt
这将把字符串 "Hello, World!" 写入到 output.txt
文件中。如果文件已存在,内容将被覆盖。如果文件还没被创建,则会自动创建并写入
2.2 重定向标准错误输出
除了标准输出,还可以将标准错误输出(stderr)重定向到文件:
cpp
echo "An error occurred" 2> error.txt
这将把标准错误输出写入到 error.txt
文件。
2.3 同时重定向标准输出和标准错误输出
有时,我们希望将标准输出和标准错误输出都重定向到同一个文件:
cpp
echo "This is normal" > output.txt echo "This is an error" 2>> output.txt
上面的命令中,标准输出被重定向到 output.txt
文件中,而标准错误输出则追加到文件末尾。
2.4 输入输出重定向与管道
重定向不仅仅用于输出,还可以用于输入。例如,我们可以使用 <
来将文件内容作为输入:
cpp
cat < input.txt
而管道(|
)则可以将一个命令的输出作为另一个命令的输入:
cpp
ls | grep "example"
这将列出当前目录下所有文件,并通过管道将输出传递给 grep
,过滤出包含 "example" 的文件名。
三、文件描述符和输出重定向的结合
上面我们所讲的输出重定向还都是最基本的输出重定向,是直接在命令行上就能实现的,下面我们来几个文件描述符的在输出重定向中的高级用法
3.1 重定向的本质
首先我们可以结合文件描述符来讲解一下输出重定向在底层中是如何进行的
首先我们先来看这样一段代码:
cpp
#include<stdio.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<string.h>
#define filename "log.txt"
int main()
{
close(1);
int fd=open(filename,O_CREAT|O_WRONLY|O_TRUNC,0666);
if(fd<0) //因为文件描述符是数组的下标,所以不可能小于0,小于0就是失败
{
perror("open file");
return 1;
}
printf("fd: %d\n",fd);
const char *msg="hello Linux\n";
int cnt=5;
while(cnt)
{
write(1,msg,strlen(msg));
cnt--;
}
close(fd);
return 0
}
运行结果:
我们将第一行的注释放开,也就是说关闭1号文件再次执行:
cpp
#include<stdio.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<string.h>
#define filename "log.txt"
int main()
{
close(1);
int fd=open(filename,O_CREAT|O_WRONLY|O_TRUNC,0666);
if(fd<0) //因为文件描述符是数组的下标,所以不可能小于0,小于0就是失败
{
perror("open file");
return 1;
}
printf("fd: %d\n",fd);
const char *msg="hello Linux\n";
int cnt=5;
while(cnt)
{
write(1,msg,strlen(msg));
cnt--;
}
close(fd);
return 0
}
运行结果:
我们比较一下上面的两段代码和输出结果:
我们可以观察到上面这两段代码唯一的区别就是一个事先把1号文件关闭了,但是它们的输出结果却有很大不同,我们的代码执行的目的就是在屏幕上打印log.txt文件的描述符,并把预期内容写入一号文件中去,而前三号文件我们在前面也讲过是我们在启动进程时是会默认创建的文件,1号文件就是显示器文件,所以第一段代码的打印结果就是直接在屏幕上输出所有打印内容,log.txt的文件标识符则为3(0,1,2被占用),而在第二段代码中,我们首先先将1号文件关闭掉,此时数组中1号文件的对应位置就是空了,此时我们在创建log.txt文件时按从小往大的顺序,就会先找到1号文件,因为1号显示器文件已经被关闭,所以往屏幕上输出内容的操作不能奏效了,而且内容也被写入在log.txt中
所以说重定向的原理就是文件描述符指向的内容的修改(个人理解)
3.2 dup2系统调用来实现重定向
首先我们先来通过man手册来看一下dup2的基本用法
cpp
man dup
我们可以看到关于dup的系统调用接口有两个,我们需要掌握的主要是第二个
我们上面的那种重定向的方法使用起来是比较繁琐的,我们必须先关闭1号文件,然后再打开新文件时才能成功,其实也可以通过dup2系统调用的方法直接实现上面的结果
来看看上面同样的代码我们如何用dup2来实现:
cpp
#include<stdio.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<string.h>
#define filename "log.txt"
int main()
{
int fd=open(filename,O_CREAT|O_WRONLY|O_TRUNC,0666);
if(fd<0) //因为文件描述符是数组的下标,所以不可能小于0,小于0就是失败
{
perror("open file");
return 1;
}
//重定向
dup2(fd,1);
printf("fd: %d\n",fd);
const char *msg="hello Linux\n";
int cnt=5;
while(cnt)
{
write(1,msg,strlen(msg));
cnt--;
}
close(fd);
return 0
}
运行结果:
3.3 其它
上面我们就讲解了输出重定向的主要使用方式,除此之外,输出重定向还可以用在几个更高级的场景,比如后台运行与输出重定向和输出重定向和多重文件描述符等,这些内容都属于拔高内容了,等我们学到后面用到的时候再讲
四、总结
文件描述符和输出重定向是 Linux 操作系统中两个非常重要的机制,它们在系统 I/O 操作、进程管理和文件系统中扮演着至关重要的角色。通过深入理解文件描述符的底层实现、相关的系统调用以及输出重定向的工作原理,我们可以更高效地与操作系统交互,了解进程之间交互的本质。
本文详细介绍了文件描述符和输出重定向的基础知识、底层实现、系统调用以及它们在 Linux 中的应用场景。希望这些知识对你有所帮助。
本篇笔记:
感谢各位大佬观看,创作不易,还请各位大佬点赞支持!!!