细说|Linux虚拟文件系统原理
文章不知道大家是否熟悉?今天我将给大家介绍《细说|Linux虚拟文件系统原理》,这篇文章主要会讲到等等知识点,如果你在看完本篇文章后,有更好的建议或者发现哪里有问题,希望大家都能积极评论指出,谢谢!希望我们能一起加油进步!
在Unix的世界里,有一句经典的话:一切对象皆是文件。这句话的含义是,可以将Unix操作系统中的所有对象都视为文件,并使用文件操作接口来处理它们。Linux作为一个类Unix操作系统,也致力于实现这一理念。
虚拟文件系统概述
为了实现“一切对象皆是文件”的理念,Linux内核引入了一个中间层:虚拟文件系统(Virtual File System)。
如果你熟悉面向对象编程语言(如C++/Java等),你可能不陌生于“接口”这个概念。虚拟文件系统类似于面向对象中的接口,定义了一套标准的接口规范。开发者只需实现这套接口,就能够通过文件操作接口来操作对象。如下图所示:

上图中的蓝色部分就是虚拟文件系统所在位置。
从上图可以看出,虚拟文件系统为上层应用提供了统一的接口。如果某个文件系统实现了虚拟文件系统的接口,那么上层应用就能够使用诸如 open()、read() 和 write() 等函数来操作它们。
今天,我们就来介绍虚拟文件系统的原理与实现。
虚拟文件系统原理
在阐述虚拟文件系统的原理前,我们先来介绍一个 Java 例子。通过这个 Java 例子,我们能够更容易理解虚拟文件系统的原理。
一个Java例子
如果大家使用过 Java 编写程序的话,那么就很容易理解虚拟文件系统了。我们使用 Java 的接口来模拟虚拟文件系统的定义:
public interface VFSFile {
int open(String file, int mode);
int read(int fd, byte[] buffer, int size);
int write(int fd, byte[] buffer, int size);
...
}
上面定义了一个名为 VFSFile 的接口,接口中定义了一些方法,如 open()、read() 和 write() 等。现在我们来定义一个名为 Ext3File 的对象来实现这个接口:
public class Ext3File implements VFSFile {
@Override
public int open(String file, int mode) {
...
}
@Override
public int read(int fd, byte[] buffer, int size) {
...
}
@Override
public int write(int fd, byte[] buffer, int size) {
...
}
...
}
现在我们就能使用 VFSFile 接口来操作 Ext3File 对象了,如下代码:
public class Main() {
public static void main(String[] args) {
VFSFile file = new Ext3File();
int fd = file.open("/tmp/file.txt", 0);
...
}
}
从上面的例子可以看出,底层对象只需要实现 VFSFile 接口,就可以使用 VFSFile 接口相关的方法来操作对象,用户完全不需要了解底层对象的实现过程。
虚拟文件系统原理
上面的 Java 例子已经大概说明虚拟文件系统的原理,但由于 Linux 是使用 C 语言来编写的,而 C 语言并没有接口这个概念。所以,Linux 内核使用了一些技巧来模拟接口这个概念。
下面来介绍一下 Linux 内核是如何实现的。
1. file结构
为了模拟接口,Linux 内核定义了一个名为 file 的结构体,其定义如下:
struct file {
...
const struct file_operations *f_op;
...
};
在 file 结构中,最为重要的一个字段就是 f_op,其类型为 file_operations 结构。而 file_operations 结构是由一组函数指针组成,其定义如下:
struct file_operations {
...
loff_t (*llseek) (struct file *, loff_t, int);
ssize_t (*read) (struct file *, char __user *, size_t, loff_t *);
ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *);
...
int (*open) (struct inode *, struct file *);
...
};
从 file_operations 结构的定义可以隐约看到接口的影子,所以可以猜想出,如果实现了 file_operations 结构中的方法,应该就能接入到虚拟文件系统中。
在 Linux 内核中,file 结构代表着一个被打开的文件。所以,只需要将 file 结构的 f_op 字段设置成不同文件系统实现好的方法集,那么就能够使用不同文件系统的功能。
这个过程在 __dentry_open() 函数中实现,如下所示:
static struct file *
__dentry_open(struct dentry *dentry,
struct vfsmount *mnt,
truct file *f,
int (*open)(struct inode *, struct file *),
const struct cred *cred)
{
...
inode = dentry->d_inode;
...
// 设置file结构的f_op字段为底层文件系统实现的方法集
f->f_op = fops_get(inode->i_fop);
...
return f;
}
设置好 file 结构的 f_op 字段后,虚拟文件系统就能够使用通用的接口来操作此文件了。调用过程如下:

2. file_operations结构
底层文件系统需要实现虚拟文件系统的接口,才能被虚拟文件系统使用。也就是说,底层文件系统需要实现 file_operations 结构中的方法集。
一般底层文件系统会在其内部定义好 file_operations 结构,并且填充好其方法集中的函数指针。如 minix文件系统 就定义了一个名为 minix_file_operations 的 file_operations 结构。其定义如下:
// 文件:fs/minix/file.c
const struct file_operations minix_file_operations = {
.llseek = generic_file_llseek,
.read = do_sync_read,
.aio_read = generic_file_aio_read,
.write = do_sync_write,
.aio_write = generic_file_aio_write,
.mmap = generic_file_mmap,
.fsync = generic_file_fsync,
.splice_read = generic_file_splice_read,
};
也就是说,如果当前使用的是 minix 文件系统,当使用 read() 函数读取其文件的内容时,那么最终将会调用 do_sync_read() 函数来读取文件的内容。
3. dentry结构
到这里,虚拟文件系统的原理基本分析完毕,但还有两个非常重要的结构要介绍一下的:dentry 和 inode。
dentry 结构表示一个打开的目录项,当我们打开文件 /usr/local/lib/libc.so 文件时,内核会为文件路径中的每个目录创建一个 dentry 结构。如下图所示:

可以看到,file 结构有个指向 dentry 结构的指针,如下所示:
struct file {
...
struct path f_path;
...
const struct file_operations *f_op;
...
};
struct path {
...
struct dentry *dentry;
};
与文件类似,目录也有相关的操作接口,所以在 dentry 结构中也有操作方法集,如下所示:
struct dentry {
...
struct dentry *d_parent; // 父目录指针
struct qstr d_name; // 目录名字
struct inode *d_inode; // 指向inode结构
...
const struct dentry_operations *d_op; // 操作方法集
...
};
其中的 d_op 字段就是目录的操作方法集。
内核在打开文件时,会为路径中的每个目录创建一个 dentry 结构,并且使用 d_parent 字段来指向其父目录项,这样就能通过 d_parent 字段来追索到根目录。
4. inode结构
在 Linux 内核中,inode 结构表示一个真实的文件。为什么有了 dentry 结构还需要 inode 结构呢?这是因为 Linux 存在硬链接的概念。
例如使用以下命令为 /usr/local/lib/libc.so 文件创建一个硬链接:
ln /usr/local/lib/libc.so /tmp/libc.so
现在 /usr/local/lib/libc.so 和 /tmp/libc.so 指向同一个文件,但它们的路径是不一样的。所以,就需要引入 inode 结构了。如下图所示:

由于 /usr/local/lib/libc.so 和 /tmp/libc.so 指向同一个文件,所以它们都使用同一个 inode 对象。
inode 结构保存了文件的所有属性值,如文件的创建时间、文件所属用户和文件的大小等。其定义如下所示:
struct inode {
...
uid_t i_uid; // 文件所属用户
gid_t i_gid; // 文件所属组
...
struct timespec i_atime; // 最后访问时间
struct timespec i_mtime; // 最后修改时间
struct timespec i_ctime; // 文件创建时间
...
unsigned short i_bytes; // 文件大小
...
const struct file_operations *i_fop; // 文件操作方法集(用于设置file结构)
...
};
我们注意到 inode 结构有个类型为 file_operations 结构的字段 i_fop,这个字段保存了文件的操作方法集。当用户调用 open() 系统调用打开文件时,内核将会使用 inode 结构的 i_fop 字段赋值给 file 结构的 f_op 字段。我们再来重温下赋值过程:
static struct file *
__dentry_open(struct dentry *dentry,
struct vfsmount *mnt,
truct file *f,
int (*open)(struct inode *, struct file *),
const struct cred *cred)
{
...
// 文件对应的inode对象
inode = dentry->d_inode;
...
// 使用inode结构的i_fop字段赋值给file结构的f_op字段
f->f_op = fops_get(inode->i_fop);
...
return f;
}
总结
本文主要介绍了 虚拟文件系统 的基本原理,从分析中可以发现,虚拟文件系统使用了类似于面向对象编程语言中的接口概念。正是有了 虚拟文件系统,Linux 才能支持各种各样的文件系统。
好了,本文到此结束,带大家了解了《细说|Linux虚拟文件系统原理》,希望本文对你有所帮助!关注golang学习网公众号,给大家分享更多文章知识!
Win10飞行模式是灰色的不能上网怎么解决 Win10飞行模式是灰色的不能上网解决办法
- 上一篇
- Win10飞行模式是灰色的不能上网怎么解决 Win10飞行模式是灰色的不能上网解决办法
- 下一篇
- 如何修复Windows Steam 中Failed to Load steamui.dll错误
-
- 文章 · linux | 11小时前 |
- Linux日志分析与故障排查技巧
- 338浏览 收藏
-
- 文章 · linux | 14小时前 |
- Linux查看CPU使用率常用命令有哪些
- 125浏览 收藏
-
- 文章 · linux | 14小时前 |
- Linux进程管理与优先级设置技巧
- 345浏览 收藏
-
- 文章 · linux | 14小时前 |
- Linux终止用户所有进程的方法
- 408浏览 收藏
-
- 文章 · linux | 17小时前 |
- Linux软件安装方法全解析
- 488浏览 收藏
-
- 文章 · linux | 17小时前 |
- Linux服务依赖管理,systemd全面解析
- 469浏览 收藏
-
- 文章 · linux | 20小时前 |
- Linux搭建DHCP服务器详细教程
- 434浏览 收藏
-
- 文章 · linux | 23小时前 |
- Linux流量监控:iftop与nload使用教程
- 354浏览 收藏
-
- 文章 · linux | 1天前 |
- Linux查看所有用户命令大全
- 140浏览 收藏
-
- 文章 · linux | 1天前 |
- Linux系统更新与补丁管理技巧
- 218浏览 收藏
-
- 文章 · linux | 1天前 |
- LinuxShell脚本入门教程指南
- 333浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
-
- ChatExcel酷表
- ChatExcel酷表是由北京大学团队打造的Excel聊天机器人,用自然语言操控表格,简化数据处理,告别繁琐操作,提升工作效率!适用于学生、上班族及政府人员。
- 3206次使用
-
- Any绘本
- 探索Any绘本(anypicturebook.com/zh),一款开源免费的AI绘本创作工具,基于Google Gemini与Flux AI模型,让您轻松创作个性化绘本。适用于家庭、教育、创作等多种场景,零门槛,高自由度,技术透明,本地可控。
- 3419次使用
-
- 可赞AI
- 可赞AI,AI驱动的办公可视化智能工具,助您轻松实现文本与可视化元素高效转化。无论是智能文档生成、多格式文本解析,还是一键生成专业图表、脑图、知识卡片,可赞AI都能让信息处理更清晰高效。覆盖数据汇报、会议纪要、内容营销等全场景,大幅提升办公效率,降低专业门槛,是您提升工作效率的得力助手。
- 3448次使用
-
- 星月写作
- 星月写作是国内首款聚焦中文网络小说创作的AI辅助工具,解决网文作者从构思到变现的全流程痛点。AI扫榜、专属模板、全链路适配,助力新人快速上手,资深作者效率倍增。
- 4557次使用
-
- MagicLight
- MagicLight.ai是全球首款叙事驱动型AI动画视频创作平台,专注于解决从故事想法到完整动画的全流程痛点。它通过自研AI模型,保障角色、风格、场景高度一致性,让零动画经验者也能高效产出专业级叙事内容。广泛适用于独立创作者、动画工作室、教育机构及企业营销,助您轻松实现创意落地与商业化。
- 3826次使用
-
- 命令行工具:应对Linux服务器安全挑战的利器
- 2023-10-04 501浏览
-
- 如何使用Docker进行容器的水平伸缩和负载均衡
- 2023-11-07 501浏览
-
- linux .profile的作用是什么
- 2024-04-07 501浏览
-
- 如何解决s权限位引发postfix及crontab异常
- 2024-11-21 501浏览
-
- 如何通过脚本自动化Linux上的K8S安装
- 2025-02-17 501浏览

