news 2026/5/6 12:02:05

Linux 文件描述符(File Descriptor, FD)详解

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Linux 文件描述符(File Descriptor, FD)详解

文件描述符(File Descriptor, FD)

文件描述符是操作系统为进程管理打开文件/资源分配的整数句柄。它是Unix/Linux系统中所有I/O操作的核心抽象

核心概念

1.本质是什么?

  • 一个非负整数(0, 1, 2, 3, …)
  • 指向内核维护的打开文件表的索引
  • 进程级别的资源标识符

2.三个标准文件描述符

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

如何工作?

内核中的数据结构关系

进程A 内核 ├── 文件描述符表 ├── 系统级打开文件表 │ fd 0 → 条目1 │ 条目1 → 文件信息1 │ fd 1 → 条目2 │ 条目2 → 文件信息2 │ fd 4 → 条目3 │ 条目3 → 文件信息3 └── └──

实际例子

// 打开文件时获取文件描述符intfd=open("file.txt",O_RDONLY);// fd 可能是 3(如果 0,1,2 已被占用)// 通过 fd 进行读写read(fd,buffer,size);write(fd,buffer,size);// 完成后关闭close(fd);

文件描述符可以指向什么?

不仅仅是文件!可以是任何I/O资源

资源类型例子如何创建
普通文件/etc/passwdopen()
目录/home/useropendir()
网络套接字TCP/UDP连接socket()
管道进程间通信pipe()
设备文件/dev/tty,/dev/nullopen()
符号链接软链接文件open()

查看文件描述符

在进程中查看

# 查看当前shell的文件描述符ls-la /proc/$$/fd# 输出示例:# lrwx------ 1 user user 64 Dec 10 10:00 0 -> /dev/pts/0# lrwx------ 1 user user 64 Dec 10 10:00 1 -> /dev/pts/0# lrwx------ 1 user user 64 Dec 10 10:00 2 -> /dev/pts/0# lr-x------ 1 user user 64 Dec 10 10:00 3 -> /etc/passwd

查看进程打开的文件描述符

# 使用 lsof 查看lsof-p<PID># 使用 /procls-la /proc/<PID>/fd/cat/proc/<PID>/fdinfo/3# 查看 fd 3 的详细信息

文件描述符的特性

1. 继承性

# 子进程继承父进程的文件描述符# 这就是为什么重定向能在管道中工作:ls-l|greptxt2>errors.log# grep 继承了 ls 的 stdout 和 stderr

2. 引用计数

// 多个描述符可以指向同一个文件intfd1=open("file.txt",O_RDONLY);// refcount = 1intfd2=dup(fd1);// refcount = 2close(fd1);// refcount = 1close(fd2);// refcount = 0 → 文件真正关闭

3. 文件描述符表 vs 打开文件表

进程A 进程B fd3 ----\-->系统打开文件表条目1 -->实际文件 fd2 ----/

实际应用场景

1. 重定向

# 将 stdout (fd 1) 重定向到文件exec1>output.logecho"这会写入文件"# 将 stderr (fd 2) 重定向到 stdoutcommand2>&1# 创建自定义文件描述符exec3<>/tmp/socket# 打开 fd 3 用于读写echo"data">&3# 写入 fd 3readline<&3# 从 fd 3 读取

2. 网络编程

// socket() 返回的也是一个文件描述符!intsock_fd=socket(AF_INET,SOCK_STREAM,0);connect(sock_fd,...);// 使用 fd 连接send(sock_fd,data,...);// 使用 fd 发送recv(sock_fd,buffer,...);// 使用 fd 接收

3. 进程间通信

intpipe_fd[2];pipe(pipe_fd);// pipe_fd[0] 用于读,pipe_fd[1] 用于写if(fork()==0){// 子进程:关闭读端,写入数据close(pipe_fd[0]);write(pipe_fd[1],"Hello",6);}

限制和配置

查看限制

# 查看单个进程最大文件描述符数ulimit-n# 查看系统全局限制cat/proc/sys/fs/file-max# 查看当前已使用的文件描述符数cat/proc/sys/fs/file-nr# 输出:已分配 已使用 最大值

修改限制

# 临时修改ulimit-n65536# 永久修改(系统级)echo"fs.file-max = 1000000">>/etc/sysctl.conf sysctl -p# 用户级限制# 编辑 /etc/security/limits.conf# * soft nofile 65536# * hard nofile 65536

常见问题

“Too many open files” 错误

# 原因:文件描述符耗尽# 查看某个进程打开的文件数ls/proc/<PID>/fd|wc-l# 解决方案:# 1. 增加系统限制# 2. 检查程序是否有文件泄露(未关闭)

文件描述符泄露检测

# 使用 lsof 监控watch-n1'lsof -p <PID> | wc -l'# 使用 /proc 监控watch-n1'ls /proc/<PID>/fd | wc -l'

总结

文件描述符的核心价值

  1. 统一抽象:所有I/O资源(文件、网络、管道)都用 fd 表示
  2. 简单接口:只需read()write()close()几个系统调用
  3. 高效管理:内核统一管理,用户态只用整数操作
  4. 资源控制:通过限制 fd 数量,控制资源使用

正是因为文件描述符的存在,Unix/Linux 才能实现"一切皆文件"的哲学,让复杂的I/O操作变得简单统一。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/4 12:19:24

基于单片机的家庭防盗报警器设计

1 系统方案设计 1.1 方案论证 防盗报警系统要求实现的功能&#xff0c;按照本设计的要求&#xff0c;可分为三个部分&#xff0c;即控制器部分、信号检测部分、报警部分。当有人经过报警器工作范围内&#xff0c;报警系统的信号检测模块会把检测到的信号转为电信号送到单片机&a…

作者头像 李华
网站建设 2026/5/5 18:35:36

6024NR14130电源模块

6024NR14130 电源模块 — ABB Taylor 系列型号: 6024NR14130 类型: 模块化电源模块功能与应用6024NR14130 是一款工业级模块化电源模块&#xff0c;主要用于 ABB Taylor 系列或 MOD300 控制系统中。它为控制系统的 CPU、I/O 模块、通信模块等提供稳定电源&#xff0c;保证整个系…

作者头像 李华
网站建设 2026/5/5 0:02:25

脸都不要了!美国开始正式出售委内瑞拉石油

当地时间1月14日&#xff0c;一位美国政府官员透露&#xff0c;美国已完成首批委内瑞拉石油的销售。 这笔交易价值5亿美元&#xff0c;该官员补充说&#xff0c;预计未来几天和几周还将进行更多石油销售&#xff0c;图2为委内瑞拉一处炼油厂。 1月3日凌晨&#xff0c;美国对委…

作者头像 李华
网站建设 2026/5/4 21:12:52

综合评估显示,这些工具在准确性和自然度方面已达到实用化水平

在众多AI论文工具中&#xff0c;选择一款适合自己需求的平台可能令人眼花缭乱。本文将对比8款热门工具&#xff0c;重点聚焦降重、降AIGC率、写论文等功能。工具排名基于实测数据和用户反馈&#xff0c;确保客观实用性。以下是简要排行表&#xff08;基于效率、准确性和易用性&…

作者头像 李华
网站建设 2026/5/5 14:25:32

订单流水分配密码:1%流水如何贯穿五层实现公平激励?

通过前五篇文章&#xff0c;我们了解了各层的权益。今天&#xff0c;我们来揭秘贯穿整个体系、让Suo有激励自动运转的“血液系统”——流水分配密码。1. 核心规则&#xff1a;1%流水分配的网状结构这套体系的一个关键设计是&#xff0c;平台总流水会按一个很小的固定比例&#…

作者头像 李华