0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

一文搞懂文件描述符

阿铭linux 来源:阿铭linux 2023-02-10 13:56 次阅读

Linux中的文件描述符

我们知道在Linux系统中一切皆可以看成是文件,文件又可分为:普通文件、目录文件、链接文件和设备文件。在操作这些所谓的文件的时候,我们每操作一次就找一次名字,这会耗费大量的时间和效率。所以Linux中规定每一个文件对应一个索引,这样要操作文件的时候,我们直接找到索引就可以对其进行操作了。

e699e7d8-a8db-11ed-bfe3-dac502259ad0.png

文件描述符(file descriptor)就是内核为了高效管理这些已经被打开的文件所创建的索引,其是一个非负整数(通常是小整数),用于指代被打开的文件,所有执行I/O操作的系统调用都通过文件描述符来实现。同时还规定系统刚刚启动的时候,0是标准输入,1是标准输出,2是标准错误。这意味着如果此时去打开一个新的文件,它的文件描述符会是3,再打开一个文件文件描述符就是4......

Linux内核对所有打开的文件有一个文件描述符表格,里面存储了每个文件描述符作为索引与一个打开文件相对应的关系,简单理解就是下图这样一个数组,文件描述符(索引)就是文件描述符表这个数组的下标,数组的内容就是指向一个个打开的文件的指针。

上面只是简单理解,实际上关于文件描述符,Linux内核维护了3个数据结构:

进程级的文件描述符表

系统级的打开文件描述符表

文件系统的i-node表

一个 Linux 进程启动后,会在内核空间中创建一个 PCB 控制块,PCB 内部有一个文件描述符表(File descriptor table),记录着当前进程所有可用的文件描述符,也即当前进程所有打开的文件。进程级的描述符表的每一条记录了单个进程所使用的文件描述符的相关信息,进程之间相互独立,一个进程使用了文件描述符3,另一个进程也可以用3。除了进程级的文件描述符表,系统还需要维护另外两张表:打开文件表、i-node 表。这两张表存储了每个打开文件的打开文件句柄(open file handle)。一个打开文件句柄存储了与一个打开文件相关的全部信息。

系统级的打开文件描述符表:

当前文件偏移量(调用read()和write()时更新,或使用lseek()直接修改)

打开文件时的标识(open()的flags参数

文件访问模式(如调用open()时所设置的只读模式、只写模式或读写模式)

信号驱动相关的设置

对该文件i-node对象的引用,即i-node 表指针

文件系统的i-node表:

文件类型(例如:常规文件、套接字或FIFO)和访问权限

一个指针,指向该文件所持有的锁列表

文件的各种属性,包括文件大小以及与不同类型操作相关的时间戳

文件描述符、打开的文件句柄以及i-node之间的关系如下图:

e6ac3960-a8db-11ed-bfe3-dac502259ad0.png

在进程 A 中,文件描述符 1 和 20 都指向了同一个打开文件表项,标号为 23(指向了打开文件表中下标为 23 的数组元素),这可能是通过调用 dup()、dup2()、fcntl() 或者对同一个文件多次调用了 open() 函数形成的。

进程 A 的文件描述符 2 和进程 B 的文件描述符 2 都指向了同一个文件,这可能是在调用 fork() 后出现的(即进程 A、B 是父子进程关系),或者是不同的进程独自去调用 open() 函数打开了同一个文件,此时进程内部的描述符正好分配到与其他进程打开该文件的描述符一样。

进程 A 的描述符 0 和进程 B 的描述符 3 分别指向不同的打开文件表项,但这些表项均指向 i-node 表的同一个条目(标号为 1976); 换言之,它们指向了同一个文件。 发生这种情况是因为每个进程各自对同一个文件发起了 open() 调用。 同一个进程两次打开同一个文件,也会发生类似情况。

这就说明:同一个进程的不同文件描述符可以指向同一个文件; 不同进程可以拥有相同的文件描述符; 不同进程的同一个文件描述符可以指向不同的文件(一般也是这样,除了 0、1、2 这三个特殊的文件); 不同进程的不同文件描述符也可以指向同一个文件。

Linux系统文件描述符示例

比如在Linux上用 vim test.py 打开一个文件,保持打开状态,再新打开一个新的shell,输入命令pidof vim 获取vim进程的pid号,然后 ll /proc/$pid/fd 查看vim 进程所使用的文件描述符列表。

e6c40630-a8db-11ed-bfe3-dac502259ad0.png

/dev/pts是远程登录(telnet,ssh等)后创建的控制台设备文件所在的目录。 因为我是通过Xshell远程登录的,所以标准输入0,标准输出1,标准错误2的文件描述符都指向虚拟终端控制台 /dev/pts/6 。 再看下面是新打开的 test.py 的文件描述符,竟然是4,说好的从3开始呢?

因为vim这种编辑器的原理是先打开源文件并拷贝,然后关闭源文件再打开自己的副本,修改完文件保存的时候直接将副本重命名覆盖源文件。 所以打开源文件的时候用的文件描述符3,然后打开自己的副本是时候就该用文件描述符4了,然后关闭源文件,文件描述符3就被释放了,我们查看的时候就只剩下了4,这里它指向的是vim创建的副本文件。

审核编辑:汤梓红

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 内核
    +关注

    关注

    3

    文章

    1309

    浏览量

    39846
  • Linux
    +关注

    关注

    87

    文章

    10990

    浏览量

    206734
  • Linux系统
    +关注

    关注

    4

    文章

    567

    浏览量

    26911
  • 文件
    +关注

    关注

    1

    文章

    540

    浏览量

    24402
  • 描述符
    +关注

    关注

    0

    文章

    3

    浏览量

    6026

原文标题:一文搞懂文件描述符

文章出处:【微信号:aming_linux,微信公众号:阿铭linux】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    请问SPI DMA描述符列表如何单次触发?

    },{ SpiTxDesptr[0], X, X,X,X}}; 其中X代表些配置信息。 请问如何数据流可实现,单次触发SPIDMA描述符个元素,手动触发三次 不断手动轮询实现。 即 手动
    发表于 01-12 08:07

    文件描述符0、1、2),(stdin、stdout、stderr),(终端设...

    前言在Linux系统中,切设备都看作文件。而每打开文件,就有个代表该打开文件
    发表于 03-02 22:05

    【OK210试用体验】【第五篇】嵌入式Linux开发学习笔记【第3课 文件描述符

    :------------------------------------------文件描述符的生成open(),open64(), creat(), creat64()socket()socketpair()pipe()与单
    发表于 08-27 22:00

    USB描述符详解

    对于实现USB功能的项目,不管是实现CDC,大容量存储、HID等都必须使用实现USB标准描述符,从而实现USB枚举过程。这里共享个USB描述符的详解。
    发表于 06-12 14:46

    类型描述符

    引用句柄的类型描述符的每个数值元素都表示什么信息,找了好多资料都没有介绍这个问题的,请高手们帮忙分析下,谢谢!
    发表于 04-30 13:04

    UBL UBOOT 描述符

       在DM368的开发包目录下/psp/flash-utils/DM36x/GNU:执行make 可以省成UBL:ubl_DM36x_nand.bin ;我想问的是如何添加UBL 的描述符?烧写到
    发表于 06-21 08:48

    请问host buff 描述符为什么没有回到return qum,而host pkt 描述符能够正常归还?

    host buff 和 host pkt 描述符设置的return qmgr 和return qnum 是样的,pkt dma 搬移后 host pkt描述符能够回到return qnum,host buff
    发表于 08-06 06:22

    zigbee描述符匹配请求方式,请问协调器可以通过什么得到其组网的终端的终端描述符

    请问下,协调器可以通过什么得到其组网的终端的终端描述符?从个终端查找终端描述符afFindEndPointDesc(byte endpoint),这个方法是指终端本身查找,还是说协
    发表于 08-09 07:35

    请问SPIDMA描述符列表如何单次触发?

    ;SpiRxDesptr[2], X, X,X,X},{&SpiTxDesptr[0], X, X,X,X}};其中X代表些配置信息。请问如何数据流可实现,单次触发SPIDMA描述符个元素,手动触发三次 不断
    发表于 10-12 09:16

    USB设备检测过程的USB设备的描述符

    。不同的描述符从不同的层级来表示设备的属性。设备描述符包含了整个设备的信息以及设备支持的配置号码,每个设备只能有个设备描述符。但每
    发表于 11-12 09:23

    传递文件描述符是什么?

    我买了本书《Linux网络编程》。 看到第11章高级套接字。有个例子,传递文件描述符。 其中,进程A根据用户输入的文件名打开
    发表于 05-07 02:18

    ARM中,Linux 文件描述符到底是什么?

    个 Linux 进程启动后,会在内核空间中创建个 PCB 控制块,PCB 内部有文件描述符表(File descriptor tab
    发表于 10-23 16:11

    以太网数据帧的发送描述符队列是什么意思?

    这个描述符是什么呀,类似linux系统的文件描述符
    发表于 05-26 06:15

    Linux中文件文件描述符概述

    6.2 Linux中文件文件描述符概述 在Linux中对目录和设备的操作都等同于文件的操作,因此,大大简化了系统对不同设备的处理,提高了效率。Linux中的
    发表于 10-18 14:35 0次下载

    Linux系统编程中的文件描述符调用

    文件描述符 进程每打开一个文件的时候,会获得该文件文件描述符,而后续的读写操作都把
    的头像 发表于 09-02 09:50 1435次阅读
    Linux系统编程中的<b class='flag-5'>文件</b><b class='flag-5'>描述符</b>调用