QT M/V架构开发实战:QFileSystemModel介绍

### 目录

  • [@[TOC](目录)](#目录 @TOC 前言 一、QFileSystemModel初步介绍 二、基本功能 1.创建 2.基本属性与方法 三、示例(简单的文件浏览器) 四、性能注意事项)
  • [前言](#目录 @TOC 前言 一、QFileSystemModel初步介绍 二、基本功能 1.创建 2.基本属性与方法 三、示例(简单的文件浏览器) 四、性能注意事项)
  • [一、QFileSystemModel初步介绍](#目录 @TOC 前言 一、QFileSystemModel初步介绍 二、基本功能 1.创建 2.基本属性与方法 三、示例(简单的文件浏览器) 四、性能注意事项)
  • [二、基本功能](#目录 @TOC 前言 一、QFileSystemModel初步介绍 二、基本功能 1.创建 2.基本属性与方法 三、示例(简单的文件浏览器) 四、性能注意事项)
  • [1.创建](#目录 @TOC 前言 一、QFileSystemModel初步介绍 二、基本功能 1.创建 2.基本属性与方法 三、示例(简单的文件浏览器) 四、性能注意事项)
  • [2.基本属性与方法](#目录 @TOC 前言 一、QFileSystemModel初步介绍 二、基本功能 1.创建 2.基本属性与方法 三、示例(简单的文件浏览器) 四、性能注意事项)
  • [三、示例(简单的文件浏览器)](#目录 @TOC 前言 一、QFileSystemModel初步介绍 二、基本功能 1.创建 2.基本属性与方法 三、示例(简单的文件浏览器) 四、性能注意事项)
  • [四、性能注意事项](#目录 @TOC 前言 一、QFileSystemModel初步介绍 二、基本功能 1.创建 2.基本属性与方法 三、示例(简单的文件浏览器) 四、性能注意事项)

前言


本文主要介绍的是使用代码生成的情况下对控件的介绍,包括拥有的功能及能修改的样式,也会说明在qtdesiner拖拽控件生成和使用代码生成控件的区别(如果有的话,遇到了的会说),此版本不属于最终版本,以后遇到什么新奇的点会继续更新!本文基于QT官方的文档进行的编写,QT版本为qt 5.14.0,编写环境为Windows11。不得不说官方文档真是个好东西,有时候有些不会的上去一看就能有灵感解决了,可惜没有中文版本的。

一、QFileSystemModel初步介绍

QFileSystemModel是 Qt 提供的一个极其强大且实用的模型类,专门用于展示和操作本地文件系统。它抽象了文件系统的层次结构(目录树),并将其完美地适配到 Qt 的 Model-View 框架中,使得在 QTreeView、QListView或 QTableView中构建文件浏览器变得非常简单。

核心思想 :​​

QFileSystemModel继承自 QAbstractItemModel(更具体地说,通常是 QDirModel的替代者,后者已废弃)。

将本地文件系统的目录结构映射为一个树状模型。

每个文件或目录对应模型中的一个项 (QModelIndex)。

提供了丰富的接口来获取文件属性(名称、路径、大小、类型、修改时间、权限、图标等)。

支持​​异步加载​​(默认),这意味着在后台线程中读取目录内容,避免阻塞 UI 线程,尤其是在处理大型目录或网络驱动器时体验更流畅。

自动监听文件系统的变化(通过 QFileSystemWatcher),并在文件或目录被添加、删除、重命名或修改时​​自动更新模型和视图​​。

读写操作(需谨慎):​​ 支持通过模型接口重命名、删除文件/目录(需要权限)。

二、基本功能

1.创建

cpp 复制代码
#include <QFileSystemModel>

QFileSystemModel *model = new QFileSystemModel(parent);

2.基本属性与方法

1)设置根路径

关键一步​​!告诉模型从哪个目录开始构建树状结构。视图(如 QTreeView)通常会自动展开并显示此根路径下的内容

cpp 复制代码
// 设置根路径为当前用户的主目录
model->setRootPath(QDir::homePath());

// 设置根路径为 C 盘 (Windows)
model->setRootPath("C:/");

// 设置根路径为根目录 (Linux/macOS)
model->setRootPath("/");

2)连接视图

cpp 复制代码
QTreeView *treeView = new QTreeView;
treeView->setModel(model); // 设置模型

// 通常,你会将视图的根索引设置为模型的根路径索引
treeView->setRootIndex(model->index(model->rootPath()));

3)获取文件信息 (通过 QModelIndex或 QFileInfo)

使用 data()函数 (指定角色)

cpp 复制代码
QModelIndex index = ...; // 例如 treeView->currentIndex() 或从选择模型获取

// 文件/目录名 (DisplayRole)
QString name = model->data(index, Qt::DisplayRole).toString();

// 完整路径 (FilePathRole - 这是 QFileSystemModel 特有的便捷角色)
QString fullPath = model->data(index, QFileSystemModel::FilePathRole).toString();
// 或者使用标准的 ToolTipRole 有时也会显示路径 (取决于视图)
QString toolTipPath = model->data(index, Qt::ToolTipRole).toString();

// 文件大小 (SizeRole) - 目录通常返回 -1 或 0
qint64 size = model->data(index, QFileSystemModel::SizeRole).toLongLong();

// 文件类型 (TypeRole) - 例如 "JPEG image", "Text Document", "Folder"
QString type = model->data(index, QFileSystemModel::TypeRole).toString();

// 最后修改时间 (LastModifiedRole)
QDateTime lastModified = model->data(index, QFileSystemModel::LastModifiedRole).toDateTime();

// 图标 (DecorationRole)
QIcon icon = model->data(index, Qt::DecorationRole).value<QIcon>();

使用 fileInfo()(更直接)

cpp 复制代码
QFileInfo fileInfo = model->fileInfo(index);
QString name = fileInfo.fileName();
QString fullPath = fileInfo.absoluteFilePath();
qint64 size = fileInfo.size();
bool isDir = fileInfo.isDir();
bool isFile = fileInfo.isFile();
QDateTime lastModified = fileInfo.lastModified();
QFile::Permissions perms = fileInfo.permissions();
// ... 使用 QFileInfo 的所有丰富接口 ...

4)过滤显示的文件 (setFilter)

使用 QDir::Filters枚举组合来指定哪些文件和目录应该被显示。QDir::NoDotAndDotDot:通常建议加上,以排除 .(当前目录) 和 ...(父目录) 项。

cpp 复制代码
// 只显示目录 (不显示文件)
model->setFilter(QDir::AllDirs | QDir::NoDotAndDotDot);

// 显示所有文件和目录 (默认)
model->setFilter(QDir::AllEntries | QDir::NoDotAndDotDot); // 通常排除 "." 和 ".."

// 显示所有文件 (包括隐藏文件) 和目录
model->setFilter(QDir::AllEntries | QDir::Hidden | QDir::NoDotAndDotDot);

// 只显示 .txt 和 .cpp 文件以及所有目录
model->setFilter(QDir::AllDirs | QDir::Files | QDir::NoDotAndDotDot);
model->setNameFilters(QStringList() << "*.txt" << "*.cpp");
model->setNameFilterDisables(false); // true: 隐藏不匹配项; false: 显示但不匹配 (通常false)

5)排序 (sort)

视图的列顺序通常是:0=名称, 1=大小, 2=类型, 3=修改日期 (取决于 headerData实现)。

cpp 复制代码
// 按名称排序 (默认)
treeView->setSortingEnabled(true); // 允许用户点击表头排序
treeView->sortByColumn(0, Qt::AscendingOrder); // 按第0列(名称)升序

// 在代码中指定排序 (例如按大小降序)
model->sort(2, Qt::DescendingOrder); // 假设大小在第2列 (需要根据视图的列定义)

6)监听目录加载完成 (directoryLoaded信号)

由于加载是异步的,如果你需要在某个目录内容加载完成后执行操作(如展开、选择特定项),可以连接此信号。

cpp 复制代码
connect(model, &QFileSystemModel::directoryLoaded,
        this, [this, model](const QString &path) {
    if (path == targetDirectoryPath) {
        // 找到并选中目标文件,或展开目录等
        QModelIndex targetIndex = model->index(targetFilePath);
        treeView->setCurrentIndex(targetIndex);
        treeView->scrollTo(targetIndex);
    }
});
model->setRootPath(targetDirectoryPath); // 或者调用 fetchMore(index) 触发加载

7)文件操作 (需谨慎!)

重命名 (setData)

cpp 复制代码
QModelIndex oldIndex = ...;
bool success = model->setData(oldIndex, "NewFileName.txt", Qt::EditRole);
// 成功会触发 dataChanged 信号,视图更新
// 失败通常是因为权限不足或文件正在使用

创建目录 (mkdir)

cpp 复制代码
QModelIndex parentDirIndex = ...; // 要在哪个目录下创建
QModelIndex newDirIndex = model->mkdir(parentDirIndex, "NewFolderName");
if (!newDirIndex.isValid()) {
    // 创建失败 (权限、重名等)
}

​删除 (remove)

cpp 复制代码
QModelIndex indexToRemove = ...;
bool success = model->remove(indexToRemove);
if (!success) {
    // 删除失败 (权限、文件不存在等)
}

注意事项:​​ 这些操作会​​直接作用于真实文件系统​​!务必进行错误处理,并考虑提供用户确认(如删除确认对话框)。对于复杂的文件操作(复制、移动),通常建议使用 QFile、QDir或 QProcess调用系统命令,而不是依赖模型的 remove或 setData。

三、示例(简单的文件浏览器)

cpp 复制代码
#include <QApplication>
#include <QSplitter>
#include <QTreeView>
#include <QListView>
#include <QFileSystemModel>

int main(int argc, char *argv[]) {
    QApplication app(argc, argv);

    // 创建分割器窗口
    QSplitter *splitter = new QSplitter;

    // 创建模型
    QFileSystemModel *model = new QFileSystemModel;
    model->setRootPath(QDir::homePath()); // 设置根路径
    model->setFilter(QDir::AllEntries | QDir::NoDotAndDotDot); // 显示所有项

    // 创建树视图 (左侧 - 目录树)
    QTreeView *treeView = new QTreeView(splitter);
    treeView->setModel(model);
    treeView->setRootIndex(model->index(model->rootPath())); // 设置视图根索引
    treeView->setSortingEnabled(true); // 允许排序
    treeView->setAnimated(true); // 展开折叠动画
    treeView->setIndentation(20); // 缩进

    // 创建列表视图 (右侧 - 当前目录内容)
    QListView *listView = new QListView(splitter);
    listView->setModel(model);
    listView->setViewMode(QListView::IconMode); // 图标模式
    listView->setIconSize(QSize(64, 64)); // 图标大小

    // 连接树视图的点击信号,更新列表视图的根索引
    QObject::connect(treeView, &QTreeView::clicked,
                     [model, listView](const QModelIndex &index) {
                         if (model->isDir(index)) {
                             listView->setRootIndex(index); // 点击目录,在列表视图中显示其内容
                         }
                     });

    // 初始设置列表视图显示根目录内容
    listView->setRootIndex(model->index(model->rootPath()));

    splitter->setWindowTitle("Simple File Browser");
    splitter->resize(800, 600);
    splitter->show();

    return app.exec();
}

四、性能注意事项

1、大型目录 :​​ 加载包含成千上万文件的目录会消耗内存和时间。考虑:

使用 QDir手动加载部分项(但这失去了 MV 自动更新的优势)。

使用 QSortFilterProxyModel进行过滤,减少显示项**。

提示用户目录很大。

2、​​网络驱动器/慢速介质:​​ 异步加载尤为重要。directoryLoaded信号有助于知道何时加载完成。

​​3、文件监控:​​ QFileSystemWatcher可能会消耗资源(尤其是在 Windows 上监控大量文件)。如果不需要实时更新,可以考虑禁用(但 QFileSystemModel内部集成,没有直接禁用选项,可能需要自定义模型)。

本次分享就到这里了,如果有什么错误的话请指正,或者有什么疑问的,也可以在评论区一起探讨!

相关推荐
kebeiovo18 分钟前
muduo网络库事件驱动模型的实现与架构
网络·架构
长路归期无望1 小时前
C语言小白实现多功能计算器的艰难历程
c语言·开发语言·数据结构·笔记·学习·算法
是大强2 小时前
stm32摇杆adc数据分析
开发语言
蓝莓味的口香糖3 小时前
【JS】什么是单例模式
开发语言·javascript·单例模式
linux kernel3 小时前
第二十三讲:特殊类和类型转换
开发语言·c++
笨蛋少年派3 小时前
JAVA基础语法
java·开发语言
渡我白衣3 小时前
深入剖析:boost::intrusive_ptr 与 std::shared_ptr 的性能边界和实现哲学
开发语言·c++·spring
爱吃小胖橘3 小时前
Lua语法
开发语言·unity·lua
怀旧,3 小时前
【C++】26. 智能指针
开发语言·c++·算法
Aevget4 小时前
PHP智能开发工具PhpStorm v2025.2全新上线——支持PHPUnit 12等
开发语言·ide·php·phpstorm