当前位置: 首页 > news >正文

Qt从入门到入土(九) -model/view(模型/视图)框架

简介

Qt的模型/视图(Model/View)架构是一种用于分离数据处理和用户界面展示的设计模式。它允许开发者将数据存储和管理(模型)与数据的显示和交互(视图)解耦,从而提高代码的可维护性和可扩展性。

Model/View的基本结构如下图所示:

各部分功能如下所示:

  • 数据(Data):表示实际的数据
  • 模型(Model):与实际数据连接,并为视图组件提供数据接口。从原始数据中提取所需内容,用于视图进行显示和编辑。
  • 视图(View):从模型中获取每个数据项的模型索引,通过模型索引获取数据,然后为界面组件提供显示数据。

在Qt的模型/视图架构中:

  • 数据模型和显示界面分离,允许同一数据模型在多个视图中展示,且可在不修改模型的情况下自定义视图。

  • 代理功能允许用户自定义数据的显示和编辑方式。在标准视图中,代理通过模型索引与数据模型通信,并提供编辑器(如QLineEdit)。

  • 模型、视图和代理通过信号和槽进行通信:

    • 数据模型变化时,通知视图更新。

    • 视图操作时,通知模型和代理。

    • 编辑数据时,代理通知模型和视图编辑器状态。

数据模型

所有基于项数据的模型都继承自QAbstractItemModel类,该类定义了视图和代理访问数据的接口。数据模型中无需直接存储数据,数据可以来自其他类、文件、数据库或其他任何数据源。

抽象类不能直接使用,需要子类继承并实现一些虚函数。Qt中包含了用于项数据处理的模型类,如下表:

Model 类用途
QStringListModel用于处理字符串列表数据的数据模型类
QStandardltemModel标准的基于项数据的数据模型类,每个项数据可以是任何数据类型
QFileSystemModel计算机上文件系统的数据模型类
QSortFilterProxyModel与其他数据模型结合,提供排序和过滤功能的数据模型类
QSqlQueryModel用于数据库SQL查询结果的数据模型类
QSqlTableModel用于数据库的一个数据表的数据模型类
QSqlRelationalTableModel用于关系型数据表的数据模型类

如果现有的这些模型类无法满足需求,可以从 QAbstractltemModel、QAbstractListModel 或 QAbstractTableModel 继承,生成自己的数据模型类。

视图组件

视图组件(View)就是显示数据模型的数据的界面组件。

Qt中提供的视图组件,如下表:

QListView用于显示单列的列表数据,适用于一维数据的操作
QTreeView用于显示树状结构数据,适用于树状结构数据的操作
QTableView用于显示表格状数据,适用于二维表格型数据的操作
QColumnView用多个QListView显示树状层次结构,树状结构的一层用一个QListView显示
QHeaderView提供行表头或列表头的视图组件,如QTableView的行表头和列表头

代理(Delegate)

在Qt中,代理(Delegate)用于在视图组件中编辑数据,负责从模型获取数据并显示在编辑器中,编辑完成后将数据保存回模型。默认情况下,Qt使用QStyledItemDelegate作为代理,它基于QAbstractItemDelegate(抽象基类)。对于特殊数据编辑需求(如整数输入使用QSpinBox,选择数据使用QComboBox),可以通过继承QStyledItemDelegate创建自定义代理。

Model/View结构

如上图,在Qt的模型/视图架构中,所有数据模型类都继承自QAbstractItemModel,并以表格的层次结构表示数据,为视图组件和代理提供统一的数据存取接口。无论底层数据结构如何组织,数据模型的表现形式可以是列表、表格或树状结构。数据模型的基本单元是项(item),每个项由行号、列号和父项定义其位置。在列表和表格中,所有项共享一个顶层项,实际顶层项并不存在;而在树状结构中,行号、列号和父项的关系更复杂,但足以唯一确定一个项的位置并存取其数据。

模型索引

为了隔离数据表示和存取方式,Qt引入了模型索引(QModelIndex)的概念。每个数据项通过模型索引进行存取,视图组件和代理通过模型索引与数据模型交互。模型索引是一个临时指针,用于提取或修改数据。由于数据模型的内部结构可能变化,模型索引是临时的。如果需要持久化的模型索引,则使用QPersistentModelIndex

获取索引

数据模型的基本形式是表格数据,但底层数据不一定以二维数组存储。行号和列号仅是为了方便组件间交互的约定。通过模型索引(QModelIndex)的行号、列号和父项索引可以存取数据。

  • 列表和表格模型:顶层节点用QModelIndex()表示,所有数据项的父项是顶层项。

// 顶层节点总是用 QModelIndex() 表示

QModelIndex index1 = model->index(0, 0, QModelIndex());

QModelIndex index2 = model->index(1, 1, QModelIndex());

  • 树状结构模型:节点可以有父节点,构造模型索引时需指定行号、列号和父节点索引。例如,节点B的父节点是节点A,则其索引通过model->index(row, column, parentIndex)生成。

// 节点1 3的父节点是顶层节点,节点2的父节点是节点1

QModelIndex index1= model->index(0, 0, QModelIndex());
QModelIndex index3 = model->index(2, 1, QModelIndex());

QModelIndex index2 = model->index(1,0,index1); 

项的角色

在Qt的数据模型中,每个项可以设置不同角色(role)的数据,用于满足不同的显示或交互需求。QStandardItemModel中的QStandardItem通过setData(const QVariant &value, int role)方法设置数据,其中role指定数据的角色,默认为Qt::UserRole + 1。常见的角色包括:

  • Qt::DisplayRole:用于视图组件中显示的字符串。

  • Qt::ToolTipRole:用于鼠标悬停时的提示信息。

  • Qt::DecorationRole:用于装饰显示(如图标)。

  • Qt::UserRole:用于自定义数据。

获取项的数据时,也需要通过role指定获取哪种角色的数据,例如data(int role)。视图组件和代理会根据角色来解释和显示数据,不同组件对角色数据的处理方式可能不同,某些角色的数据也可能被忽略。

模型视图的基本使用

创建视图

视图就是一个窗口,所以使用模型/视图框架时无需使用QWidget,可以直接调用视图内置显示函数用于显示

// 创建视图
QListView view;

// 把模型交给视图显示
view.setModel(model);
view.show();

创建模型

// 创建模型
auto model = new QStandardItemModel(&a);

添加项

// 给模型添加数据项
model->appendRow(new QStandardItem("肖战"));
model->insertRow(0,new QStandardItem("蔡徐坤"));

添加子项

给模型中的QStandardItem添加子项,只对QTreeView起作用

QStandardItem* item = new QStandardItem("蔡徐坤");
                                                   
auto* subItem = new QStandardItem("个人信息");          
subItem->appendRow(new QStandardItem("18岁"));    
subItem->appendRow(new QStandardItem("身高190cm"));    
                                                   
item->appendRow(subItem);                          
item->appendRow(new QStandardItem("生活习惯"));         
                                                   
model->appendRow(item);

修改项

// 修改数据项
model->setItem(1,new QStandardItem("陈立农"));

model->setData(index,"大碗宽面",Qt::ItemDataRole::DisplayRole);   //也可用于修改数据项

删除项

// 删除数据项,从模型中移除并释放内存
//model->removeRow(0);
//model->removeRows(0,2);   // 指定起始行和行数
// 删除数据项,但不释放内存
auto itemLists = model->takeRow(1);
qDebug()<<itemLists.size();
for(auto &item:itemLists)
{
    // 需要手动释放内存
    delete item;
}
qDebug()<<model->rowCount()<<" "<<model->columnCount();

获取项的索引

// item索引
auto item = new QStandardItem("wyf");
model->appendRow(item);
QModelIndex index = model->indexFromItem(item);
qDebug()<<index;

查找项

// 查询数据项
auto res = model->findItems("蔡徐坤");
if(res.isEmpty())
    qDebug()<<"未找到该数据!";
for(auto& r:res)
{
    qDebug()<<"找到数据:"<<r->data(Qt::ItemDataRole::DisplayRole).toString();
}

设置角色数据

// 角色(ItemDataRole)
model->setData(index,"大碗宽面",Qt::ItemDataRole::DisplayRole);   //也可用于修改数据项
item->setData(QColor(0,0,0),Qt::ItemDataRole::DecorationRole);   //设置装饰角色
item->setData("我是帅哥",Qt::ItemDataRole::UserRole);             //设置用户角色
qDebug()<<item->data(Qt::UserRole).toString();                //获取用户角色信息 

视图信号连接槽

// 连接信号与槽
QObject::connect(&view,&QListView::clicked,[=](const QModelIndex& idx){
      qDebug()<<idx.data(Qt::DisplayRole).toString();
});

综合代码

#include <QStandardItemModel>
#include <QListView>
#include <QApplication>int main(int argc, char *argv[])
{QApplication a(argc, argv);// 创建模型auto model = new QStandardItemModel(&a);// 创建视图QListView view;// 把模型交给视图显示view.setModel(model);view.show();// 给模型添加数据项model->appendRow(new QStandardItem("肖战"));model->insertRow(0,new QStandardItem("蔡徐坤"));// 修改数据项model->setItem(1,new QStandardItem("陈立农"));// 删除数据项,从模型中移除并释放内存//model->removeRow(0);//model->removeRows(0,2);   // 指定起始行和行数// 删除数据项,但不释放内存auto itemLists = model->takeRow(1);qDebug()<<itemLists.size();for(auto &item:itemLists){// 需要手动释放内存delete item;}qDebug()<<model->rowCount()<<" "<<model->columnCount();// item索引auto item = new QStandardItem("wyf");model->appendRow(item);QModelIndex index = model->indexFromItem(item);qDebug()<<index;// 角色(ItemDataRole)model->setData(index,"大碗宽面",Qt::ItemDataRole::DisplayRole);   //也可用于修改数据项item->setData(QColor(0,0,0),Qt::ItemDataRole::DecorationRole);   //设置装饰角色item->setData("我是帅哥",Qt::ItemDataRole::UserRole);             //设置用户角色qDebug()<<item->data(Qt::UserRole).toString();                //获取用户角色信息// 查询数据项auto res = model->findItems("蔡徐坤");if(res.isEmpty())qDebug()<<"未找到该数据!";for(auto& r:res){qDebug()<<"找到数据:"<<r->data(Qt::ItemDataRole::DisplayRole).toString();}// 设置是否可选item->setCheckable(true);// 连接信号与槽QObject::connect(&view,&QListView::clicked,[=](const QModelIndex& idx){qDebug()<<idx.data(Qt::DisplayRole).toString();});// 添加子项,只对QTreeView视图起作用// auto subItem = new QStandardItem("个人信息");// subItem->appendRow(new QStandardItem("18岁"));// subItem->appendRow(new QStandardItem("身高180cm"));// model->item(0)->appendRow(subItem);return a.exec();
}

相关文章:

Qt从入门到入土(九) -model/view(模型/视图)框架

简介 Qt的模型/视图&#xff08;Model/View&#xff09;架构是一种用于分离数据处理和用户界面展示的设计模式。它允许开发者将数据存储和管理&#xff08;模型&#xff09;与数据的显示和交互&#xff08;视图&#xff09;解耦&#xff0c;从而提高代码的可维护性和可扩展性。…...

缓存之美:Guava Cache 相比于 Caffeine 差在哪里?

大家好&#xff0c;我是 方圆。本文将结合 Guava Cache 的源码来分析它的实现原理&#xff0c;并阐述它相比于 Caffeine Cache 在性能上的劣势。为了让大家对 Guava Cache 理解起来更容易&#xff0c;我们还是在开篇介绍它的原理&#xff1a; Guava Cache 通过分段&#xff08;…...

[漏洞篇]XSS漏洞详解

[漏洞篇]XSS漏洞 一、 介绍 概念 XSS&#xff1a;通过JS达到攻击效果 XSS全称跨站脚本(Cross Site Scripting)&#xff0c;为避免与层叠样式表(Cascading Style Sheets, CSS)的缩写混淆&#xff0c;故缩写为XSS。这是一种将任意 Javascript 代码插入到其他Web用户页面里执行以…...

【Leetcode 每日一题】2269. 找到一个数字的 K 美丽值

问题背景 一个整数 n u m num num 的 k k k 美丽值定义为 n u m num num 中符合以下条件的 子字符串 数目&#xff1a; 子字符串长度为 k k k。子字符串能整除 n u m num num。 给你整数 n u m num num 和 k k k&#xff0c;请你返回 n u m num num 的 k k k 美丽值…...

IO进程线程(线程)

作业 1.创建两个线程&#xff0c;分支线程1拷贝文件的前一部分&#xff0c;分支线程2拷贝文件的后一部分 2.创建三个线程&#xff0c;实现线程A打印A&#xff0c;线程B打印B&#xff0c;线程C打印C&#xff1b;重复打印顺序ABC。 信号量实现&#xff1a; 条件变量实现&#x…...

1-002:MySQL InnoDB引擎中的聚簇索引和非聚簇索引有什么区别?

在 MySQL InnoDB 存储引擎 中&#xff0c;索引主要分为 聚簇索引&#xff08;Clustered Index&#xff09; 和 非聚簇索引&#xff08;Secondary Index&#xff09;。它们的主要区别如下&#xff1a; 1. 聚簇索引&#xff08;Clustered Index&#xff09; 定义 聚簇索引是表数…...

tomcat单机多实例部署

一、部署方法 多实例可以运行多个不同的应用&#xff0c;也可以运行相同的应用&#xff0c;类似于虚拟主机&#xff0c;但是他可以做负载均衡。 方式一&#xff1a; 把tomcat的主目录挨个复制&#xff0c;然后把每台主机的端口给改掉就行了。 优点是最简单最直接&#xff0c;…...

论文阅读分享——UMDF(AAAI-24)

概述 题目&#xff1a;A Unified Self-Distillation Framework for Multimodal Sentiment Analysis with Uncertain Missing Modalities 发表&#xff1a;The Thirty-Eighth AAAI Conference on Artificial Intelligence (AAAI-24) 年份&#xff1a;2024 Github&#xff1a;暂…...

解决asp.net mvc发布到iis下安全问题

解决asp.net mvc发布到iis下安全问题 环境信息1.The web/application server is leaking version information via the "Server" HTTP response2.确保您的Web服务器、应用程序服务器、负载均衡器等已配置为强制执行Strict-Transport-Security。3.在HTML提交表单中找不…...

概念|RabbitMQ 消息生命周期 待消费的消息和待应答的消息有什么区别

目录 消息生命周期 一、消息创建与发布阶段 二、消息路由与存储阶段 三、消息存活与过期阶段 四、消息投递与消费阶段 五、消息生命周期终止 关键配置建议 待消费的消息和待应答的消息 一、待消费的消息&#xff08;Unconsumed Messages&#xff09; 二、待应答的消息…...

springboot三层架构详细讲解

目录 springBoot三层架构 0.简介1.各层架构 1.1 Controller层1.2 Service层1.3 ServiceImpl1.4 Mapper1.5 Entity1.6 Mapper.xml 2.各层之间的联系 2.1 Controller 与 Service2.2 Service 与 ServiceImpl2.3 Service 与 Mapper2.4 Mapper 与 Mapper.xml2.5 Service 与 Entity2…...

2025最新群智能优化算法:云漂移优化(Cloud Drift Optimization,CDO)算法求解23个经典函数测试集,MATLAB

一、云漂移优化算法 云漂移优化&#xff08;Cloud Drift Optimization&#xff0c;CDO&#xff09;算法是2025年提出的一种受自然现象启发的元启发式算法&#xff0c;它模拟云在大气中漂移的动态行为来解决复杂的优化问题。云在大气中受到各种大气力的影响&#xff0c;其粒子的…...

2025年Draw.io最新版本下载安装教程,附详细图文

2025年Draw.io最新版本下载安装教程&#xff0c;附详细图文 大家好&#xff0c;今天给大家介绍一款非常实用的流程图绘制软件——Draw.io。不管你是平时需要设计流程图、绘制思维导图&#xff0c;还是制作架构图&#xff0c;甚至是简单的草图&#xff0c;它都能帮你轻松搞定。…...

记录--洛谷 P1451 求细胞数量

如果想查看完整题目&#xff0c;请前往洛谷 P1451 求细胞数量 P1451 求细胞数量 题目描述 一矩形阵列由数字 0 0 0 到 9 9 9 组成&#xff0c;数字 1 1 1 到 9 9 9 代表细胞&#xff0c;细胞的定义为沿细胞数字上下左右若还是细胞数字则为同一细胞&#xff0c;求给定矩形…...

Android Studio 配置国内镜像源

Android Studio版本号&#xff1a;2022.1.1 Patch 2 1、配置gradle国内镜像&#xff0c;用腾讯云 镜像源地址&#xff1a;https\://mirrors.cloud.tencent.com/gradle 2、配置Android SDK国内镜像 地址&#xff1a;Index of /AndroidSDK/...

做到哪一步才算精通SQL

做到哪一步才算精通SQL-Structured Query Language 数据定义语言 DDL for StructCREATE&#xff1a;用来创建数据库、表、索引等对象ALTER&#xff1a;用来修改已存在的数据库对象DROP&#xff1a;用来删除整个数据库或者数据库中的表TRUNCATE&#xff1a;用来删除表中所有的行…...

Manus演示案例: 英伟达财务估值建模 解锁投资洞察的深度剖析

在当今瞬息万变的金融投资领域&#xff0c;精准剖析企业价值是投资者决胜市场的关键。英伟达&#xff08;NVIDIA&#xff09;&#xff0c;作为科技行业的耀眼明星&#xff0c;其在人工智能和半导体领域的卓越表现备受瞩目。Manus 凭借专业的财务估值建模能力&#xff0c;深入挖…...

postman接口请求中的 Raw是什么

前言 在现代的网络开发中&#xff0c;API 的使用已经成为数据交换的核心方式之一。然而&#xff0c;在与 API 打交道时&#xff0c;关于如何发送请求体&#xff08;body&#xff09;内容类型的问题常常困扰着开发者们&#xff0c;尤其是“raw”和“json”这两个术语之间的区别…...

DeepSeek大语言模型下几个常用术语

昨天刷B站看到复旦赵斌老师说的一句话“科幻电影里在人脑中植入芯片或许在当下无法实现&#xff0c;但当下可以借助AI人工智能实现人类第二脑”&#xff08;大概是这个意思&#xff09; &#x1f49e;更多内容&#xff0c;可关注公众号“ 一名程序媛 ”&#xff0c;我们一起从 …...

ctf-WEB: 关于 GHCTF Message in a Bottle plus 与 Message in a Bottle 的非官方wp解法

Message in a Bottle from bottle import Bottle, request, template, runapp Bottle()# 存储留言的列表 messages [] def handle_message(message):message_items "".join([f"""<div class"message-card"><div class"me…...

使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式

一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明&#xff1a;假设每台服务器已…...

未来机器人的大脑:如何用神经网络模拟器实现更智能的决策?

编辑&#xff1a;陈萍萍的公主一点人工一点智能 未来机器人的大脑&#xff1a;如何用神经网络模拟器实现更智能的决策&#xff1f;RWM通过双自回归机制有效解决了复合误差、部分可观测性和随机动力学等关键挑战&#xff0c;在不依赖领域特定归纳偏见的条件下实现了卓越的预测准…...

渗透实战PortSwigger靶场-XSS Lab 14:大多数标签和属性被阻止

<script>标签被拦截 我们需要把全部可用的 tag 和 event 进行暴力破解 XSS cheat sheet&#xff1a; https://portswigger.net/web-security/cross-site-scripting/cheat-sheet 通过爆破发现body可以用 再把全部 events 放进去爆破 这些 event 全部可用 <body onres…...

UDP(Echoserver)

网络命令 Ping 命令 检测网络是否连通 使用方法: ping -c 次数 网址ping -c 3 www.baidu.comnetstat 命令 netstat 是一个用来查看网络状态的重要工具. 语法&#xff1a;netstat [选项] 功能&#xff1a;查看网络状态 常用选项&#xff1a; n 拒绝显示别名&#…...

多模态商品数据接口:融合图像、语音与文字的下一代商品详情体验

一、多模态商品数据接口的技术架构 &#xff08;一&#xff09;多模态数据融合引擎 跨模态语义对齐 通过Transformer架构实现图像、语音、文字的语义关联。例如&#xff0c;当用户上传一张“蓝色连衣裙”的图片时&#xff0c;接口可自动提取图像中的颜色&#xff08;RGB值&…...

【Go】3、Go语言进阶与依赖管理

前言 本系列文章参考自稀土掘金上的 【字节内部课】公开课&#xff0c;做自我学习总结整理。 Go语言并发编程 Go语言原生支持并发编程&#xff0c;它的核心机制是 Goroutine 协程、Channel 通道&#xff0c;并基于CSP&#xff08;Communicating Sequential Processes&#xff0…...

ElasticSearch搜索引擎之倒排索引及其底层算法

文章目录 一、搜索引擎1、什么是搜索引擎?2、搜索引擎的分类3、常用的搜索引擎4、搜索引擎的特点二、倒排索引1、简介2、为什么倒排索引不用B+树1.创建时间长,文件大。2.其次,树深,IO次数可怕。3.索引可能会失效。4.精准度差。三. 倒排索引四、算法1、Term Index的算法2、 …...

LLM基础1_语言模型如何处理文本

基于GitHub项目&#xff1a;https://github.com/datawhalechina/llms-from-scratch-cn 工具介绍 tiktoken&#xff1a;OpenAI开发的专业"分词器" torch&#xff1a;Facebook开发的强力计算引擎&#xff0c;相当于超级计算器 理解词嵌入&#xff1a;给词语画"…...

用docker来安装部署freeswitch记录

今天刚才测试一个callcenter的项目&#xff0c;所以尝试安装freeswitch 1、使用轩辕镜像 - 中国开发者首选的专业 Docker 镜像加速服务平台 编辑下面/etc/docker/daemon.json文件为 {"registry-mirrors": ["https://docker.xuanyuan.me"] }同时可以进入轩…...

Pinocchio 库详解及其在足式机器人上的应用

Pinocchio 库详解及其在足式机器人上的应用 Pinocchio (Pinocchio is not only a nose) 是一个开源的 C 库&#xff0c;专门用于快速计算机器人模型的正向运动学、逆向运动学、雅可比矩阵、动力学和动力学导数。它主要关注效率和准确性&#xff0c;并提供了一个通用的框架&…...