以下是对您提供的技术博文进行深度润色与专业重构后的版本。我以一位有十年Qt工业UI开发经验的工程师视角,彻底摒弃模板化表达、AI腔调和教科书式结构,转而采用真实项目中的思考脉络 + 现场调试口吻 + 工程权衡细节来重写全文。语言更紧凑、逻辑更锋利、案例更扎心,同时严格遵循您提出的全部格式与风格要求(无“引言/总结/展望”等标题、不使用机械连接词、禁用空洞术语堆砌、关键点加粗提示、代码注释直击痛点)。
QListView排序不是“配个代理就行”,是模型契约的落地实战
上周在客户现场调一个告警列表卡顿问题,用户点击“严重等级”列后要等800ms才刷新——不是数据量大(才327条),而是他们在QSortFilterProxyModel::lessThan()里写了三行QSqlQuery去查设备归属部门……这已经不是Qt问题了,是对模型层职责的根本误读。
QListView本身从不排序。它只问模型:“第5行该显示什么?” 模型答:“你拿这个QModelIndex来找我。” ——排序这件事,从来就该由模型或它的代理来回答,而不是让视图去猜、让线程去等、让用户去忍。
真正可靠的自定义排序,必须同时满足三件事:
-不破坏MVC边界:视图不碰业务规则,模型不耦合UI线程;
-扛得住实时插入:新告警进来时,不能全量重排;
-改起来像改配置:运营提“把‘已确认’状态排到最后”,开发不该重编译。
下面拆解两个真实踩过坑的方案——不是理论对比,是你明天就能粘贴进项目的硬核路径。
代理模型不是“套壳”,是索引重映射的精密手术
QSortFilterProxyModel最常被误解成“给模型加个滤镜”。其实它干的是件更底层的事:维护一张源模型索引到视图索引的翻译表。你看到的“排序后列表”,本质是一张view_row → source_row的哈希映射,lessThan()只是这张表的生成器。
所以别在lessThan()里做任何耗时操作。我们曾在线上环境发现有人在里面调QFile::exists()检查日志文件是否存在——单次比较慢12ms,1000条数据排序直接卡死GUI线程。lessThan()必须是纯函数:输入两个QModelIndex,输出true/false,中间不能有任何副作用,不能访问外部状态,不能触发信号。
来看一个电力SCADA系统的真实实现:
class AlarmSortProxy : public QSortFilterProxyModel {