【Qt之QSqlRelationalTableModel】描述及使用

news2024/11/25 13:31:57

描述

QSqlRelationalDelegate链接: https://blog.csdn.net/MrHHHHHH/article/details/134690139

QSqlRelationalTableModel类为单个数据库表提供了一个可编辑的数据模型,并支持外键。
QSqlRelationalTableModel的行为类似于QSqlTableModel,但允许将列设置为其他数据库表的外键。

左边的屏幕截图显示了QTableView中一个普通的QSqlTableModel。外键(城市和国家)不能解析为人类可读的值。
右边的屏幕截图显示了一个QSqlRelationalTableModel,外键被解析为人类可读的文本字符串。

下面的代码片段展示了如何建立QSqlRelationalTableModel:在这里插入图片描述

	  model->setTable("employee");
      model->setRelation(2, QSqlRelation("city", "id", "name"));
      model->setRelation(3, QSqlRelation("country", "id", "name"));

setRelation()函数调用在两个表之间建立关系。第一个调用指定表employee中的第2列是一个外键,它映射表city的字段id,并且视图应该向用户显示城市的名称字段。第二个调用对列3执行类似的操作。

如果使用读写的QSqlRelationalTableModel,可能希望在视图上使用QSqlRelationalDelegate。与默认委托不同,QSqlRelationalDelegate为作为其他表的外键的字段提供了一个组合框。要使用该类,只需在带有QSqlRelationalDelegate实例的视图上调用QAbstractItemView::setItemDelegate():

	  QTableView *view = new QTableView;
      view->setModel(model);
      view->setItemDelegate(new QSqlRelationalDelegate(view));

relationaltablemodel示例演示了如何将QSqlRelationalTableModelQSqlRelationalDelegate结合使用,为表提供外键支持。
在这里插入图片描述
注意:

  • 表必须声明主键。
  • 表的主键可能不包含对另一个表的关系。
  • 如果关系表包含引用引用表中不存在的行的键,则包含无效键的行将不通过模型公开。用户或数据库有责任维护引用完整性。
  • 如果关系的显示列名称也用作关系表中的列名称,或者如果它在多个关系中用作显示列名称,则会被别名。别名是关系的表名、显示列名和一个唯一的id,用下划线连接(例如tablename_columnname_id)。QSqlRecord::fieldName()将返回别名列名。当检测到重复时,所有重复显示列名的出现都会被别名,但不会对主表中的列名进行任何别名操作。别名对QSqlRelation没有影响,因此QSqlRelation::displayColumn()将返回原始显示列名。
  • 参考表名称被别名。别名是单词"relTblAl"和关联的列索引用下划线连接(例如,relTblAl_2)。别名可用于过滤表(例如,setFilter("relTblAl_2='Oslo' OR relTblAl_3='USA'"))。
  • 使用setData()时,角色应始终为Qt::EditRole,使用data()时,角色应始终为Qt::DisplayRole

常用方法

  1. 枚举:enum JoinMode
常量描述解释
QSqlRelationalTableModel::InnerJoin0- Inner join mode, return rows when there is at least one match in both tables.— 内部连接模式,当两个表中至少有一个匹配时返回行。
QSqlRelationalTableModel::InnerJoin1- Left join mode, returns all rows from the left table (table_name1), even if there are no matches in the right table (table_name2).— 左连接模式,返回左表(table_name1)中的所有行,即使右表(table_name2)中没有匹配。
  1. void setJoinMode(QSqlRelationalTableModel::JoinMode joinMode)

设置SQL joinMode以显示或隐藏外键为NULL的行。在InnerJoin模式(默认)中,这些行将不会显示:如果想显示它们,请使用LeftJoin模式。

  1. QSqlRelation relation(int column) const

返回列的关系,如果没有设置关系,则返回无效关系。

  1. QSqlTableModel* relationModel(int column) const

返回一个QSqlTableModel对象,用于访问列为外键的表,如果给定列没有关系,则返回0。
返回的对象归QSqlRelationalTableModel所有。

  1. void setRelation(int column, const QSqlRelation &relation)

让指定的列是由关系指定的外索引。
Ex:

    model->setTable("employee");
    model->setRelation(2, QSqlRelation("city", "id", "name"));

setRelation()调用指定employee表中的第2列是一个外键,它映射表city的字段id,并且视图应该向用户显示城市的名称字段。
注意:表的主键不能包含与另一个表的关系。

  1. void clear()

重新实现QSqlQueryModel::clear()方法。用于清除模型数据。

  1. QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const

为索引引用的项返回存储在给定角色下的数据。
注意:如果没有要返回的值,则返回无效的QVariant,而不是返回0`。

  1. bool removeColumns(int column, int count, const QModelIndex &parent = QModelIndex())

在支持此功能的模型上,从模型中删除从parent下给定列开始的计数列。
如果列已成功删除,则返回true;否则返回false。
基类实现不做任何事情并返回false。
如果实现了自己的模型,如果希望支持删除,则可以重新实现此函数。或者,可以提供自己的API来修改数据。

  1. bool select()

使用通过setTable()设置的表中的数据填充模型,使用指定的筛选和排序条件,如果成功则返回true;否则返回false。
注意:调用select()将恢复所有未提交的更改并删除所有插入的列。

  1. bool setData(const QModelIndex &index, const QVariant &value, int role = Qt::EditRole)

QAbstractItemModel::setData()重新实现。
将具有指定索引的项中角色的数据设置为给定的值。根据编辑策略的不同,值可以立即应用于数据库,也可以缓存在模型中。
如果值可以设置,则返回true,如果出现错误(例如,如果索引超出边界)则返回false。
对于关系列,value必须是索引,而不是显示值。索引也必须存在于被引用的表中,否则函数返回false。

示例

.pro中添加QT += sql
包含头文件

#include <QtWidgets>
#include <QtSql>

#include <QMessageBox>
#include <QSqlDatabase>
#include <QSqlError>
#include <QSqlQuery>
  1. // 创建表及插入数据
static bool createConnection()
{
    QSqlDatabase db = QSqlDatabase::addDatabase("QSQLITE");
    // 创建在内存中
    db.setDatabaseName(":memory:");
    if (!db.open()) {
        QMessageBox::critical(0, qApp->tr("Cannot open database"),
            qApp->tr("Unable to establish a database connection.\n"
                     "This example needs SQLite support. Please read "
                     "the Qt SQL driver documentation for information how "
                     "to build it.\n\n"
                     "Click Cancel to exit."), QMessageBox::Cancel);
        return false;
    }

    // 使用SqlQuery进行表创建及数据添加
    QSqlQuery query;
    query.exec("create table person (id int primary key, "
               "firstname text, lastname int)");
    query.exec("insert into person values(0, '0', 0)");
    query.exec("insert into person values(1, '1', 1)");
    query.exec("insert into person values(4, '2', 2)");
    query.exec("insert into person values(3, '3', 3)");

    query.exec("create table itemF (id int primary key,"
                                             "firstName varchar(20))");
    query.exec("insert into itemF "
               "values(0, '小明')");
    query.exec("insert into itemF "
               "values(2, '小华')");
    query.exec("insert into itemF "
               "values(3, '小芳')");
    query.exec("insert into itemF "
               "values(1, '小诸葛')");

    query.exec("create table itemL (id int, lastName varchar(20))");
    query.exec("insert into itemL values(0, '赵')");
    query.exec("insert into itemL values(1, '钱')");
    query.exec("insert into itemL values(2, '孙')");
    query.exec("insert into itemL values(3, '李')");

    return true;
}
  1. // 初始化模型
void initializeModel(QSqlRelationalTableModel *model)
{
    // 设置表名
    model->setTable("person");
    // 手动提交
    model->setEditStrategy(QSqlTableModel::OnManualSubmit);
    // 设置外键
    model->setRelation(1, QSqlRelation("itemF", "id", "firstName"));
    model->setRelation(2, QSqlRelation("itemL", "id", "lastName"));
    // 添加表头
    model->setHeaderData(0, Qt::Horizontal, QObject::tr("id"));
    model->setHeaderData(1, Qt::Horizontal, QObject::tr("firstName"));
    model->setHeaderData(2, Qt::Horizontal, QObject::tr("lastName"));
    // 刷新
    model->select();
}

上述代码:

  • 创建了一个名为 person 的表,并添加了一些数据。
  • 创建了两个名为 itemFitemL 的表,并添加了一些具有 idfirstNamelastName 属性的数据
  • 这些表和数据将用于后面的 QSqlRelationalTableModel 示例
  • 最后,该函数返回 true
  1. // 创建视图
QTableView *createView(const QString &title, QSqlTableModel *model)
{
    QTableView *view = new QTableView;
    // 设置模型
    view->setModel(model);
    // 设置委托,这样可以下拉框显示
    view->setItemDelegate(new QSqlRelationalDelegate(view));
    view->setWindowTitle(title);
    return view;
}

上述代码:

  • 定义了一个名为 createView 的函数,函数用于创建一个新的 QTableView
  • 函数使用一个标题和一个 QSqlTableModel 对象作为参数,并使用传入的 QSqlTableModel 对象设置 view 的模型。
  • 使用 QSqlRelationalDelegate 设置 view 的委托,以便在需要显示下拉列表时使用。
  • 最后,函数设置 view 的窗口标题,并将 view 返回给调用者。
  1. // 调用
int main(int argc, char *argv[])
{
    QApplication app(argc, argv);
    if (!createConnection())
        return 1;

    QSqlRelationalTableModel model;

    initializeModel(&model);

    QTableView *view = createView(QObject::tr("Relational Table Model"), &model);
    view->show();

    return app.exec();
}

结果

在这里插入图片描述

注意

  • 表必须声明主键
  • 使用setRelation()方法时,关联的表和外键需存在,不然,错误的行不会加载到模型中,不会在视图中显示
  • 使用setItemDelegate(new QSqlRelationalDelegate(view))时,可以将单元格以下拉框方式显示

结论

重剑无锋,大巧不工

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1267733.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

解锁Jira本地部署的数据中心版高级功能,打造高效、智能、精细化的项目管理

近日&#xff0c;在龙智携手Atlassian与JFrog共同举办的“大规模开发创新&#xff1a;如何提升企业级开发效率与质量”的线下研讨会中&#xff0c;龙智高级咨询顾问、Atlassian认证专家叶燕秀为大家带来了精彩演讲&#xff0c;解锁Jira Data Center版的诸多高级功能&#xff0c…

RLHF:强化学习结合大预言模型的训练方式

RLHF (Reinforcement Learning from Human Feedback) 以强化学习方式依据人类反馈优化语言模型。 文章目录 一、简介二、一般的流程三、微调gpt介绍示例 参考文章 一、简介 强化学习从人类反馈中学习&#xff08;RLHF&#xff0c;Reinforcement Learning from Human Feedback&a…

6 Redis缓存设计与性能优化

缓存穿透 缓存穿透是指查询一个根本不存在的数据&#xff0c; 缓存层和存储层都不会命中&#xff0c; 通常出于容错的考虑&#xff0c; 如果从存储层查不到数据则不写入缓存层。缓存穿透将导致不存在的数据每次请求都要到存储层去查询&#xff0c; 失去了缓存保护后端存储的意义…

Linux CentOS7 安装Docker | 中文输入法

CentOS7安装中文输入法&#xff1a; 在安装CentOS时&#xff0c;我们为了方便使用&#xff0c;语言选择了中文&#xff0c;但是我们发现&#xff0c;在Linux命令行或者是浏览器中输入时&#xff0c;我们只能输入英文&#xff0c;无法输入汉字。 用yum 安装ibus 命令&#xff…

05 # 基本类型

类型注解 作用&#xff1a;相当于强类型语言中的类型声明 语法&#xff1a; (变量/函数): type 原始类型: let bool: boolean true; let num: number 313; let str: string kaimo;数组: let arr: number[] [1, 2, 3]; let arr2: Array<number | string> [1, 2,…

【数据库】基于索引的扫描算法,不同类型索引下的选择与连接操作,不同的代价及优化

基于索引的算法 ​专栏内容&#xff1a; 手写数据库toadb 本专栏主要介绍如何从零开发&#xff0c;开发的步骤&#xff0c;以及开发过程中的涉及的原理&#xff0c;遇到的问题等&#xff0c;让大家能跟上并且可以一起开发&#xff0c;让每个需要的人成为参与者。 本专栏会定期更…

WordPress更改文章分类插件

当WP网站内容比较多的时候&#xff0c;有时候如果涉及到批量修改文章分类&#xff0c;如果一个个的去操作的话就太费事了&#xff0c;如果使用后台批量修改分类的话是增加旧分类不会取消选择&#xff0c;就就导致我们适得其反还需要一个一个的去编辑取消&#xff0c;实在繁琐了…

机器人与3D视觉 Robotics Toolbox Python 一 安装 Robotics Toolbox Python

一 安装python 库 前置条件需要 Python > 3.6&#xff0c;使用pip 安装 pip install roboticstoolbox-python测试安装是否成功 import roboticstoolbox as rtb print(rtb.__version__)输出结果 二 Robotics Toolbox Python样例程序 加载机器人模型 加载由URDF文件定义…

Pinia仓库统一管理

pinia独立维护 在src/stores文件夹下创建index.js文件&#xff0c;将main.js中关于pinia的语句放到index.js中 index.js文件内容&#xff1a; import { createPinia } from pinia import piniaPluginPersistedstate from pinia-plugin-persistedstate const pinia createPi…

电商API接口的接入|京东商品API接口接入说明

京东联盟商品接口API申请&#xff1a; 一、京东联盟 - 注册/登录 1、网址&#xff1a;https://union.jd.com 2、首次登录请先注册&#xff0c;注册成功后即可登录&#xff0c;需要完善个人信息&#xff0c;用于佣金结算。如果有京东商城的账号&#xff0c;也可以直接登录。 …

【存储】blotdb的原理及实现(2)

【存储】etcd的存储是如何实现的(3)-blotdb 在etcd系列中&#xff0c;我们对作为etcd底层kv存储的boltdb进行了比较全面的介绍。但是还有两个点没有涉及。 第一点是boltdb如何和磁盘文件交互。 持久化存储和我们一般业务应用程序的最大区别就是其强依赖磁盘文件。一方面文件数…

想要更快的文件传输?看看这些aspera的替代方案吧

随着数据量的增大&#xff0c;文件传输已经成为许多公司和组织日常工作中必不可少的环节之一。而对于大容量、海量数据的传输&#xff0c;普通的传输方式可能甚至无法胜任。Aspera作为一种高效的文件传输协议应运而生&#xff0c;其能够在处理大容量、高速传输方面表现出色。然…

Excel导出操作

<div class"right"> <el-button size"mini" click"exportEmployee">excel导出</el-button></div>安装file-saver $ npm i file-saver $ yarn add file-saver //下包后引入 import FileSaver from "file-sav…

开放式耳机怎么选?自费千元测评,百元、千元价位选哪个

开放式耳机以其不入耳式设计&#xff0c;更容易带给用户舒适的佩戴体验&#xff0c;也不影响使用中聆听周围声响&#xff0c;还可以保证长时间的舒适佩戴&#xff0c;适配漫长的通勤、游玩旅程。当然&#xff0c;开放式耳机种类也有许多&#xff0c;究竟哪一款更适合大家呢&…

离散系统的频域分析(数字信号处理实验2-1)

创建具有15 Hz和40 Hz分量频率的信号&#xff0c;叠加两个信号形成混合信号x&#xff0c;使用fft命令绘制x的频域图&#xff0c;标注频率为横坐标&#xff0c;平均能量为纵坐标。 文章目录 一.题目二.实验目的三.实验仪器四.实验原理1.MATLAB使用函数2.离散傅里叶变换(DFT)实验…

人工智能(pytorch)搭建模型21-基于pytorch搭建卷积神经网络VoVNetV2模型,并利用简单数据进行快速训练

大家好&#xff0c;我是微学AI&#xff0c;今天给大家介绍一下人工智能(pytorch)搭建模型21-基于pytorch搭建卷积神经网络VoVNetV2模型&#xff0c;并利用简单数据进行快速训练。VoVNetV2模型是计算机视觉领域的一个重要研究成果&#xff0c;它采用了Voice of Visual Residual&…

“2024上海智博会、2024北京智博会”双展联动,3月上海,6月北京

“2024上海智博会、2024北京智博会”双展联动&#xff0c;将分别于3月和6月在上海和北京举办。这两个展会旨在充分展示智慧城市、人工智能、物联网、大数据、软件等新兴行业的最新产品和技术。 作为中国最具影响力和创新力的智能科技展会&#xff0c;上海智博会和北京智博会吸引…

C#——Delegate(委托)与Event(事件)

C#——Delegate&#xff08;委托&#xff09;与Event&#xff08;事件&#xff09; 前言一、Delegate&#xff08;委托&#xff09;1.是什么&#xff1f;2.怎么用&#xff1f;Example 1&#xff1a;无输入无返回值Example 2&#xff1a;有输入Example 3&#xff1a;有返回值Exa…

实现数据一体化的有效措施

数据一体化的趋势 数据一体化是指将企业内部和外部的各种数据资源整合、统一管理和共享利用的过程&#xff0c;是当前信息化发展的重要趋势。 统一数据平台&#xff1a;企业越来越意识到数据是重要资产&#xff0c;需要建立统一的数据平台&#xff0c;集中管理和利用各类数据资…

直击广州车展 | 远航汽车“卷”出新高度

第23届广州车展作为2023年汽车行业的年度收官之作&#xff0c;成为各大汽车厂商“秀肌肉”的绝佳舞台&#xff0c;22万平方米的开放展区内容纳了1132辆展车&#xff0c;包括全球首发车59辆、概念车20辆、新能源车469辆。 新能源汽车产业发展迅猛&#xff0c;得益于新能源车型在…