浅析文件描述符fd和FILE结构体

文件描述符fd(file descriptor)
它在Linux编程中有着非常重要的作用!因为Linux下一切皆文件,所有对设备和文件的操作都是用文件描述符来进行。
那么什么是文件描述符呢?
它是一个非负的整数,它是一个索引值,指向内核中每个进程打开文件的记录表。当打开一个现存文件或者一个新文件时(open),内核就向进程返回一个文件描述符用于后续对文件的读写操作,即进程通过PCB中文件描述符表找到该fd所指向的文件指针file。因此在Linux系统下面,文件描述符主要用来标识一个文件。

内核通过文件对象表来管理系统中各种各样的文件,而文件表则是通过指针来指向打开的文件,进而达到管理整个文件系统的目的。

通常情况下,将一个程序从硬盘加载到内存后,这个程序就化身为了一个进程,这时系统会默认打开三个文件:
标准输入(stdin)、标准输出(stdout)、标准错误(stderr)。
这三个文件相对应的三个文件描述符分别为0、1、2。所以后面如果创建新文件,那么此时这个新文件的文件描述符就是3。
这是因为在Linux中,文件的描述符分配是从小到大逐个查询文件描述符是否已经使用,然后再分配。
相应的,如果你提前关闭了文件描述符1,那么新建的文件的描述符就是1。
文件描述符与对应的文件的关系:
这里写图片描述
FILE结构体
c语言的stdio.h头文件中,定义了用于文件操作的结构体FILE。因此我们通过fopen返回一个文件指针(指向FILE结构体的指针)来进行文件操作。文件指针指向进程的用户空间中的FILE结构体。
FILE结构体的定义:
FILE结构体中最重要的两个成员变量是:文件描述符和缓冲区的大小

//C语言文件指针域文件描述符之间可以相互转换
int fileno(FILE * stream)
FILE * fdopen(int fd, const char * mode)

struct _iobuf {
    char *_ptr;          //缓冲区当前指针
    int   _cnt;
    char *_base;         //缓冲区基址
    int   _flag;         //文件读写模式
    int   _file;         //文件描述符
    int   _charbuf;      //缓冲区剩余自己个数
    int   _bufsiz;       //缓冲区大小
    char *_tmpfname;
};
typedef struct _iobuf FILE;


c程序用不同的FILE结构体管理每个文件。程序员可以使用文件,但是不需要知道FILE结构体的细节。实际上,FILE结构体是间接地操作系统的文件控制块(FCB)来实现对文件的操作的。
FILE结构体中的_file ,也就时文件描述符,作为进入打开文件表索引的整数。

fd 和 FILE 的区别与联系:
文件描述符在:系统调用层 (sys call)
文件指针在 :c 库 (lib call)

封装体现在:
(1)缓冲方式:操作系统(sys)一般采用的是无缓冲,而lib一般则采用的是全缓冲的方式。
(2)函数封装:在lib层所对应的标准输入std_in,标准输出std_out,标准错误std_err也正好对应着文件描述符的0,1,2。
小结:
file_struct是操作系统用来管理文件的数据结构,
当我们创建一个进程时,会创建文件描述符表,
进程控制块PCB中的fs指针指向文件描述符表,
当我们创建文件时,会为指向该文件的指针FILE*关联一个文件描述符并添加在文件描述符表中。
在文件描述符表中fd相当于数组的索引,FILE*相当于数组的内容,指向一个文件结构体。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值