Linux之文件描述符

        对于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);
}

        调用结果:

        ​​​​​​​        ​​​​​​​         

        好了,以上便是一些关于文件描述符的内容,觉得还行的话,请帅哥美女们点点赞呐。😭

 

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值