0

0

Linux 文件系统从磁盘到应用的完整路径

冷漠man

冷漠man

发布时间:2026-01-30 21:59:51

|

471人浏览过

|

来源于php中文网

原创

ext4通过页缓存优先响应read(),未命中时由inode/extent树定位逻辑块,经bio提交至块层,最终由NVMe驱动转为Submission Queue命令;O_DIRECT跳过页缓存但需内存对齐,路径解析依赖dentry/inode缓存。

linux 文件系统从磁盘到应用的完整路径

ext4 文件系统如何把 read() 调用变成磁盘读取

Linux 应用调用 read() 时,不会直接发命令给硬盘。它走的是内核抽象层:先查页缓存(page cache),命中就直接返回;没命中才触发底层 I/O。ext4 作为文件系统驱动,负责把“读文件第 1024 字节”翻译成“读物理块设备上某几个 block”,而这些 block 的位置由 ext4 的 inode、间接块、extent 树共同决定。

关键点在于:ext4 不管理扇区地址,只管逻辑块号(logical block number);真正映射到磁盘 LBA 是由块设备层(如 sd 驱动)和 SCSI/ATA 协议完成的。

  • ext4 的 inode 中记录文件大小、权限、以及指向数据块的指针(直接块、间接块或 extent)
  • 小文件常用直接块;大文件倾向用 extent(连续块范围),减少元数据开销
  • read() 触发 generic_file_read_iter()ext4_readpage()mpage_readpages() → 最终提交 bio 到块层

bio 到 NVMe SSD 的实际写入路径

当 ext4 准备好要读的逻辑块列表后,会构造一个或多个 bio 结构体,交给通用块层(blk-mq)。这时还没碰硬件——bio 会被排队、合并、限速、加密(如果启用了 dm-crypt),再经由设备队列送到驱动。

对 NVMe 设备来说,最终调用的是 nvme_submit_cmd(),把命令放进 Submission Queue,由 SSD 控制器自己拉取执行。注意:NVMe 协议绕过了传统 IDE/SCSI 的中间层,所以延迟更低,但调试时也更难抓到“中间状态”。

  • 同一块数据可能被 page cachebuffer cache(已弱化)、SSD 内部 DRAM 缓存NAND 闪存的 page buffer 多次缓存
  • hdparm -I /dev/nvme0n1 可查控制器是否启用写缓存(Write Cache),这直接影响 fsync() 是否真落盘
  • 使用 blktrace + btt 可跟踪一个 read() 在块层各阶段耗时,定位卡点在调度器、驱动还是设备

为什么 open("/path/to/file", O_DIRECT) 会跳过页缓存

O_DIRECT 意味着应用告诉内核:“别管我的内存是不是对齐、别用你的 page cache,我来负责缓冲,你直通块层。” 这要求用户空间缓冲区地址和长度都按 logical_block_size 对齐(通常是 512B 或 4K),否则 read() 返回 -EINVAL

汉潮社区团购系统
汉潮社区团购系统

汉潮社区团购拼团系统以社区/农村/商业区基本单位,通过招募社区团长,通过微信群等社交工具进行开团销售,把相同一社区人群的日常所需商品交由平台+商家+团长+平台配送员集中管理运营的一种新型社区消费模式,为您提供一套完整的社区团购运营方案,帮助您快速构建运营管理体系,降低前期投入成本。系统从用户体验到供应链管理模块环环相扣,简单易懂,让您轻松玩转社区团购/拼团!安装步骤:一、配置数据库文件,修改数据库

下载

跳过页缓存看似更快,实则风险不少:没有缓存复用、每次都是真实 I/O、且容易因对齐失败静默降级为普通读(取决于内核版本和文件系统)。

  • ext4 在挂载时若指定 dioread_nolock,可避免 O_DIRECT 读时对 inode 加锁,提升并发性能
  • io_uring 配合 O_DIRECT 是当前高性能 I/O 的主流组合,但需检查内核是否开启 CONFIG_IO_URING
  • 不要在数据库日志写入等强一致性场景盲目禁用页缓存——page cache 提供了统一的脏页回写和 fsync 语义

用户态程序看到的路径名怎么变成磁盘地址

路径解析不是一次性动作。每次 open()stat() 都要从根目录开始 walk:/proc/1234/fd/0 → 找到 dentry → 关联 inode → 确认文件类型和权限 → 最终拿到 file 结构体。这个过程大量依赖 dentry cache 和 inode cache,否则每 open 一次都要读磁盘上的目录块。

ext4 的目录项(struct ext4_dir_entry_2)是变长结构,按 hash 分桶(htree 目录),所以大目录下 ls 快,但 find . -name "*.log" 仍要遍历所有叶子块。

  • ls -l 显示的 inode 号来自 ext4 的 inode table,不是磁盘物理地址;同一个 inode 号在不同挂载点可能指向不同内容(bind mount)
  • /proc/mounts 中的 uuidfstype 决定了内核用哪个文件系统驱动处理该设备
  • 硬链接共用一个 inode,软链接则是独立 inode 存放路径字符串——这意味着 readlink() 不触发数据块读取,只读 inode 自身
真正复杂的不是路径有多长,而是每一层缓存策略、锁粒度、异步时机都不一样。比如 page cache 回写由 writeback 内核线程控制,而 ext4 journal 提交又依赖 jbd2 线程,两者节奏错开就可能造成意外延迟。这类细节不出问题时没人注意,一出就是疑难杂症。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

320

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

212

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1502

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

625

2023.11.24

java读取文件转成字符串的方法
java读取文件转成字符串的方法

Java8引入了新的文件I/O API,使用java.nio.file.Files类读取文件内容更加方便。对于较旧版本的Java,可以使用java.io.FileReader和java.io.BufferedReader来读取文件。在这些方法中,你需要将文件路径替换为你的实际文件路径,并且可能需要处理可能的IOException异常。想了解更多java的相关内容,可以阅读本专题下面的文章。

655

2024.03.22

php中定义字符串的方式
php中定义字符串的方式

php中定义字符串的方式:单引号;双引号;heredoc语法等等。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

610

2024.04.29

go语言字符串相关教程
go语言字符串相关教程

本专题整合了go语言字符串相关教程,阅读专题下面的文章了解更多详细内容。

172

2025.07.29

c++字符串相关教程
c++字符串相关教程

本专题整合了c++字符串相关教程,阅读专题下面的文章了解更多详细内容。

83

2025.08.07

C++ 设计模式与软件架构
C++ 设计模式与软件架构

本专题深入讲解 C++ 中的常见设计模式与架构优化,包括单例模式、工厂模式、观察者模式、策略模式、命令模式等,结合实际案例展示如何在 C++ 项目中应用这些模式提升代码可维护性与扩展性。通过案例分析,帮助开发者掌握 如何运用设计模式构建高质量的软件架构,提升系统的灵活性与可扩展性。

14

2026.01.30

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 8.1万人学习

Git 教程
Git 教程

共21课时 | 3.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号