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

Linux 系统文件描述符(File Descriptor)小白级介绍

1. 概述

Linux 遵循"一切皆文件"的理念。在 Linux 系统中,文件描述符是一个索引值(非负整数),指向内核为每个进程所维护的该进程打开文件的记录表。

如上所述,每个进程都维护着一张文件描述符表。
文件描述符是进程级别的资源,并且文件描述符在进程内是唯一的。
另外,当某个进程结束时,其打开的所有文件描述符都会被关闭

默认情况下,一个进程启动时会自动打开 3 个文件描述符:

  • 0: 标准输入(stdin)
  • 1: 标准输出(stdout)
  • 2: 标准错误(stderr)

其他打开的文件描述符,根据不同的场景会有不同。常见的文件描述符的使用场景:如下

  • 普通文件的读写操作
  • 设备文件的访问(如终端设备)
  • 网络套接字(Socket)连接
  • 管道(Pipe)通信
  • 目录的访问
  • 符号链接的访问

2. 子进程继承父进程的文件描述符

当使用 fork() 创建子进程时,子进程会获得父进程文件描述符表的副本。

进程A (父进程)                          进程B (子进程)
+-------------------+                +-------------------+
| File Descriptor   |     fork       | File Descriptor   |
| Table (副本1)      | ------------>  | Table (副本2)     |
+-------------------+                +-------------------+↓                                  ↓↓                                  ↓都指向相同的文件表项↓↓
+-------------------+
|     File Table    |  (系统级别,共享)
|                   |
| offset, status    |
| pipe buffer       |
+-------------------+

这意味着:

  • 子进程能访问父进程打开的所有文件
  • 父子进程共享文件偏移量(file offset)
  • 共享文件状态标志(file status flags)
  • 实际上是共享了底层的文件表项

这样设计的优势在于:

  • 允许父子进程进行通信(通过读写相同文件)
  • 可以共享已打开的资源
  • 适用于管道通信等场景

但其潜在问题是:

  • 可能造成父子进程写入冲突
  • 需要注意文件描述符的正确关闭
  • 可能导致资源泄露

3. 线程和所属进程的文件描述符

线程是完全使用所属进程的文件描述符(同一个)

附:线程和子进程的区别

线程(Thread)子进程(child process)
概念是在同一个进程内的执行单元
  • LWP: Light Weight Process
  • 有自己的线程 ID(TID)
  • 有自己的栈空间
  • 有自己的寄存器上下文
是独立的进程
内存共享进程的内存空间有独立的内存空间
资源共享进程的所有资源
(包括文件描述符,和所属进程是同一个)
资源是父进程的副本
开销创建开销较小创建开销较大
通信直接通过共享内存通信,速度快需要通过IPC机制(如管道、信号等)通信
独立性一个线程崩溃可能导致整个进程崩溃一个子进程崩溃不会影响父进程
调试所有线程共享同一个地址空间,调试相对复杂各进程独立,调试相对简单
适用场景共享数据、密集交互的并发任务需要隔离的并发任务
http://www.lryc.cn/news/498178.html

相关文章:

  • 【Verilog】实验二 数据选择器的设计与vivado集成开发环境
  • IDL学习笔记(三)OMI数据处理。hdf5文件读取,图像反转,GeoTiff区别,月季年均值计算提取输出,单位转换,运行时间计算
  • 深入浅出:PHP中的数据类型全解析
  • 要使用 OpenResty 创建一个接口,返回客户端的 IP 地址,并以 JSON 格式输出
  • 智慧油客:从初识、再识OceanBase,到全栈上线
  • ClickHouse守护进程
  • 智能合约
  • SQL面试题——拼多多SQL面试题 求连续段的起始位置和结束位置
  • 玩《三角洲行动》遇到游戏运行故障是什么原因?游戏运行故障要怎么解决?预防游戏运行故障问题出现
  • 基于灰色神经网络的订单需求预测
  • 记录学习《手动学习深度学习》这本书的笔记(三)
  • JS中递归函数的理解及展开运算符在递归种的运用理解
  • 人工智能学习用的电脑安装cuda、torch、conda等软件,版本的选择以及多版本切换
  • 提高身份证 OCR 识别 API 接口的准确性的方法
  • PHP面向对象
  • Tomcat新手成长之路:安装部署优化全解析(下)
  • GPT 1到4代的演进笔记
  • vitepress组件库文档项目 markdown语法大全(修正版)
  • Vue3技术开发,使用纯CSS3动手制作一个3D环绕的相册展示效果,支持传入任意图片.3D轮播相册的组件
  • LeetCode 力扣 热题 100道(十五)搜索插入位置(C++)
  • 【035】基于51单片机俄罗斯方块游戏机【Proteus仿真+Keil程序+报告+原理图】
  • NAT traversal 原理 | TCP / UDP/ P2P
  • 如何成长为一名工程技术经理
  • GEE开发之下载海拔、坡度、坡向数据
  • gozero项目迁移与新服务器环境配置,包含服务器安装包括go版本,Nginx,项目配置包括Mysql,redis,rabbit,域名
  • Scala正则表达式全面教程
  • 伺服电机为什么会变慢?
  • 61 基于单片机的小车雷达避障及阈值可调
  • 微信小程序之手机归属地查询
  • ElementUI 问题清单