对于LInux的文件描述符,这是一个即简单又复杂的问题,简单的来说,所谓的文件描述符其实就是一个指针数组的下标。可当我们简单描述了文件描述符之后,问题就随之复杂了。即这个指针数组保存着什么样的指针?这个指针指向的是什么内容?等等问题,下面我们根据此图来浅浅探讨一下:
首先,对于系统来说,解决诸多管理的问题,一直遵循着一个原则,即“先描述再组织”。什么是先描述?即对于一个问题,我们先提炼出其属性,描述出它的大致情况。当描述完成后,便可以进行组织整合,将一个大问题转化为系统的一个小组件。
那么再从右往左看图片,赫然便是“先描述再组织”的生动写照,对于文件,我们先构建file对象进行描述,再构建files_struct将这些文件对象组织起来管理,最后用一个指针返回files_struct到进程结构体中,最终就像形成了一本花名册,交给了管理者。
所以了解了这个过程,对于开始提出的问题,是不是迎刃而解?这个指针数组保存的是file类型的指针,指向的是对文件进行描述的file结构体。
可讲到这里,你是不是觉得好像懂了些什么,又好像什么都没懂?没错,因为关于文件描述符的底层逻辑看起来就是这么简单。但真的这么简单吗?往往有些东西,侧面才能理解的更深刻!
一、Linux进程默认情况下会有3个缺省打开的文件描述符
这三个文件描述符分别是标准输入0,标准输出1,标准错误2.
0,1,2对应的物理设备一般是:键盘,显示器,显示器。
所以,对于输入输出行为,我们调用系统接口还可以这样:
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <string.h>
int main()
{
char buf[1024];
ssize_t s = read(0, buf, sizeof(buf));
if(s > 0){
buf[s] = 0;
write(1, buf, strlen(buf));
write(2, buf, strlen(buf));
}
return 0;
}
运行结果:
如图,可以看到所谓的printf与scanf等输入输出函数,不过是封装了如上代码。既然如此,你们是不是有了一些奇思妙想?那么,不要急哦,当我们了解了下面的这个知识点后,奇思妙想就能实现了。😀
二、文件描述符的分配规则
先看代码:
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
int main()
{
close(0);
int fd = open("myfile", O_RDONLY);
if(fd < 0){
perror("open");
return 1;
}
printf("fd: %d\n", fd);
close(fd);
return 0;
}
运行结果:
可以看到,当关闭0号文件描述符后,再打开一个文件,该文件分配的也是0号文件描述符,也就是说,文件描述符的分配是一种遍历式的分配,即从数组头开始,当遇到到空位置时便分配该位置的下标。
三、输入输出的重定向
好了,知道了文件描述符的分配规则,也知道了输入输出函数的实现内容,那么,这两个组合起来,是不是就能实现输入输出的重定向了?上代码:
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdlib.h>
int main()
{
close(1);
int fd = open("test_io.txt", O_WRONLY|O_CREAT, 00644);
if(fd < 0){
perror("open");
return 1;
}
printf("fd: %d\n", fd);
fflush(stdout);
close(fd);
exit(0);
}
运行结果:
看到没有,这便是输出的重定向,则输入的重定向同理。不过不管你有没有看懂,我还是来讲一下过程。
如图,第一步,关闭1号文件描述符,即将其置空,第二步,打开新的文件,因为1号文件描述符为空,那么根据分配规则,新打开的文件将被分配一号文件描述符。第三步,调用printf()输出,前面讲过,因为printf是对write(1, xxx,xxxxx)的封装,所以可以看到printf使用的文件描述符是默认为1的,只向1号文件描述符输出。那么如此,printf输出的内容便输出到了test_io.txt中,因为test_io.txt的文件描述符为1。
这听起来有一些的绕,于是为了方便起见,系统提供了一个用来重定向的函数,如下
#include <unistd.h>
int dup2(int oldfd, int newfd);
使用如下:
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdlib.h>
int main()
{
int fd = open("test_io.txt", O_WRONLY|O_CREAT, 00644);
if(fd < 0){
perror("open");
return 1;
}
close(1);
dup2(fd,1);
printf("我已经被重定向至文件\n");
fflush(stdout);
close(fd);
exit(0);
}
调用结果:
好了,以上便是一些关于文件描述符的内容,觉得还行的话,请帅哥美女们点点赞呐。😭