当前位置: 首页 > news >正文

Linux 文件描述符

Linux 文件描述符

Linux 中一切皆文件,比如 C++ 源文件、视频文件、Shell脚本、可执行文件等,就连键盘、显示器、鼠标等硬件设备也都是文件。

一个 Linux 进程可以打开成百上千个文件,为了表示和区分已经打开的文件,Linux 会给每个文件分配一个编号(一个 ID),这个编号就是一个整数,被称为文件描述符(File Descriptor)。

文件描述符(FD: File Descriptor)

在 Linux 操作系统中,当一个应用程序或进程操作系统资源时,首先会触发文件调用事件,通过应用程序或进程发送一个调用文件的信号给操作系统内核内核会在 打开文件表 中增加一条记录,同时将打开文件表中新增的记录返回给应用程序或进程,而应用程序或进程接收到来自内核返回的信息,称为 文件描述符(file desrciption)

当我们打开文件时,操作系统在内存中要**创建相应的数据结构来描述目标文件file struct,表示一个已经打开的文件对象**)。而进程执行open系统调用,必须让进程和文件关联起来。每个进程都有一个指针 *files 保存在进程控制块PCB中 , 它指向一张表 files_struct

files_struct表包含一个指针数组,每个元素都是一个指向打开文件的指针。本质上,文件描述符就是该数组的下标。所以,只要获取到文件描述符,就可以找到对应的文件。

img

文件描述符是一个简单的非负整数,用来表明每一个被进程锁打开的文件,实际上,通过 内核 返回的文件描述符正是 打开文件表 中该文件的索引记录,是一个非负整数。文件描述符 通常从 3 开始记录。

在Linux操作系统上,每个进程都有三个标准的 「POSIX」文件描述符,对应三个标准流,按照索引以 0 开始,分别是 Standard Input(标准输入)Standard Output(标准输出)Standard Error(标准错误)

文件调用流程

一个 Linux 进程启动后,会在内核空间中创建一个 PCB 控制块,PCB 内部有一个文件描述符表(File descriptor table)记录着当前进程所有可用的文件描述符,也即当前进程所有打开的文件

除了文件描述符表,系统还需要维护另外两张表:

  • 打开文件表(Open file table)
  • i-node 表(i-node table)

文件描述符表每个进程都有一个,打开文件表和 i-node 表整个系统只有一个,它们三者之间的关系如下图所示:

img

通过文件描述符,可以找到文件指针,从而进入打开文件表。该表存储了以下信息:

  • 文件偏移量,也就是文件内部指针偏移量。调用 read() 或者 write() 函数时,文件偏移量会自动更新,当然也可以使用 lseek() 直接修改。
  • 状态标志,比如只读模式、读写模式、追加模式、覆盖模式等。
  • i-node 表指针

然而,要想真正读写文件,还得通过打开文件表的 i-node 指针进入 i-node 表,该表包含了诸如以下的信息:

  • 文件类型,例如常规文件、套接字或 FIFO。
  • 文件大小
  • 时间戳,比如创建时间、更新时间。
  • 文件锁

对上图的进一步说明:

  • 在进程 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 这三个特殊的文件);
  • 不同进程的不同文件描述符也可以指向同一个文件。

参考文献

[1]: Linux文件描述符到底是什么?

[2]: 什么是文件描述符?

[3]: 文件描述符

http://www.lryc.cn/news/60548.html

相关文章:

  • 第17章_反射机制
  • 使用VBA小程序提高资产清查效率
  • JavaSE学习进阶day07_02 异常
  • 操作系统学习笔记
  • 【Spring Boot】SpringBoot设计了哪些可拓展的机制?
  • 《程序员面试金典(第6版)》面试题 10.10. 数字流的秩
  • 智能洗地机好用吗?值得入手的洗地机推荐
  • Spring Security实战(一)——基于内存和数据库模型的认证与授权
  • 轻松掌握FFmpeg编程:从架构到实践
  • 桌面应用程序开发攻略(初步了解)
  • 【李老师云计算】HBase+Zookeeper部署及Maven访问(HBase集群实验)
  • 第11章_常用类和基础API
  • Java语言数据类型与c语言数据类型的不同
  • C# Replace()、Trim()、Split()、Substring()、IndexOf() 、 LastIndexOf()函数
  • C++类的理解与类型名,类的成员,两种定义方式,类的访问限定符,成员访问,作用域与实例化对象
  • 【华为OD机试真题 C++】1051 - 处理器问题 | 机试题+算法思路+考点+代码解析
  • Linux 常用操作命令大全
  • Git使用教程
  • substrate中打印调试信息的多种方式详解
  • Disentangled Graph Collaborative Filtering
  • Nginx快速上手
  • 【设计模式】实际场景解释策略模式与工厂模式的应用
  • 外包干了三年,算是废了...
  • 九龙证券|光模块概念股封单资金超3亿元,传媒板块涨停潮来袭
  • [ES6] 数组
  • 【问题描述】编写一个程序计算出球、圆柱和圆锥的表面积和体积。
  • Python 人工智能:16~20
  • 【华为OD机试真题】最优资源分配(javapython)
  • git的使用——操作流程
  • Ae:自动定向