【Qt之QSqlRelationalTableModel】描述及使用
描述
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
示例演示了如何将QSqlRelationalTableModel
与QSqlRelationalDelegate
结合使用,为表提供外键支持。
注意:
- 表必须声明主键。
- 表的主键可能不包含对另一个表的关系。
- 如果关系表包含引用引用表中不存在的行的键,则包含无效键的行将不通过模型公开。用户或数据库有责任维护引用完整性。
- 如果关系的显示列名称也用作关系表中的列名称,或者如果它在多个关系中用作显示列名称,则会被别名。别名是关系的表名、显示列名和一个唯一的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
。
常用方法
- 枚举:
enum JoinMode
常量 | 值 | 描述 | 解释 |
---|---|---|---|
QSqlRelationalTableModel::InnerJoin | 0 | - Inner join mode, return rows when there is at least one match in both tables. | — 内部连接模式,当两个表中至少有一个匹配时返回行。 |
QSqlRelationalTableModel::InnerJoin | 1 | - 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)中没有匹配。 |
void setJoinMode(QSqlRelationalTableModel::JoinMode joinMode)
设置SQL joinMode
以显示或隐藏外键为NULL
的行。在InnerJoin
模式(默认)中,这些行将不会显示:如果想显示它们,请使用LeftJoin
模式。
QSqlRelation relation(int column) const
返回列的关系,如果没有设置关系,则返回无效关系。
QSqlTableModel* relationModel(int column) const
返回一个QSqlTableModel
对象,用于访问列为外键的表,如果给定列没有关系,则返回0。
返回的对象归QSqlRelationalTableModel
所有。
void setRelation(int column, const QSqlRelation &relation)
让指定的列是由关系指定的外索引。
Ex:
model->setTable("employee");model->setRelation(2, QSqlRelation("city", "id", "name"));
setRelation()
调用指定employee
表中的第2列是一个外键,它映射表city
的字段id
,并且视图应该向用户显示城市的名称字段。
注意:表的主键不能包含与另一个表的关系。
void clear()
重新实现QSqlQueryModel::clear()
方法。用于清除模型数据。
QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const
为索引引用的项返回存储在给定角色下的数据。
注意:如果没有要返回的值,则返回无效的QVariant,而不是返回
0`。
bool removeColumns(int column, int count, const QModelIndex &parent = QModelIndex())
在支持此功能的模型上,从模型中删除从parent
下给定列开始的计数列。
如果列已成功删除,则返回true;否则返回false。
基类实现不做任何事情并返回false。
如果实现了自己的模型,如果希望支持删除,则可以重新实现此函数。或者,可以提供自己的API来修改数据。
bool select()
使用通过setTable()
设置的表中的数据填充模型,使用指定的筛选和排序条件,如果成功则返回true;否则返回false。
注意:调用select()
将恢复所有未提交的更改并删除所有插入的列。
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>
- // 创建表及插入数据
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;
}
- // 初始化模型
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
的表,并添加了一些数据。 - 创建了两个名为
itemF
和itemL
的表,并添加了一些具有id
和firstName
或lastName
属性的数据 - 这些表和数据将用于后面的
QSqlRelationalTableModel
示例 - 最后,该函数返回
true
。
- // 创建视图
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
返回给调用者。
- // 调用
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))
时,可以将单元格以下拉框方式显示
结论
重剑无锋,大巧不工
。
相关文章:

【Qt之QSqlRelationalTableModel】描述及使用
描述 QSqlRelationalDelegate链接: https://blog.csdn.net/MrHHHHHH/article/details/134690139 QSqlRelationalTableModel类为单个数据库表提供了一个可编辑的数据模型,并支持外键。 QSqlRelationalTableModel的行为类似于QSqlTableModel,但允许将列设…...

【Openstack Train安装】四、MariaDB/RabbitMQ 安装
本章介绍了MariaDB/RabbitMQ的安装步骤,MariaDB/RabbitMQ仅需要在控制节点安装。 在安装MariaDB/RabbitMQ前,请确保您按照以下教程进行了相关配置: 【Openstack Train安装】一、虚拟机创建 【Openstack Train安装】二、NTP安装 【Opensta…...

工业级路由器在智能交通系统(ITS)中的创新应用
智能交通系统(ITS)作为一种先进的交通管理与控制系统,旨在提高交通运输系统的效率、安全性和便捷性。随着科技的不断发展,智能交通系统已经成为城市交通管理的重要组成部分。而工业级路由器作为一种可靠的网络通信设备,…...

React立即更新DOM
正常情况下,react会等待set完毕后再进行页面渲染,所以在set时无法拿到更新后的dom import { useRef, useState } from "react"export default () > {const div useRef(null)const [count, setCount] useState(0)const btnClick () >…...

[JavaScript前端开发及实例教程]计算器井字棋游戏的实现
计算器(网页内实现效果) HTML部分 <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>My Calculator&l…...
数据结构 / 队列 / 循环队列 / 结构体定义和创建
1. 结构体定义 //head.h#ifndef __QUEUE_HEAD_H__ #define __QUEUE_HEAD_H__#include <string.h> #include <stdlib.h> #include <stdio.h>#define MAXSIZE 5 //循环队列最多元素个数typedef char datatype; //数据元素类型typedef struct {datatype data[M…...

Java零基础——Redis篇
1.【熟悉】NoSQL的简介 1.1 什么是NoSQL NoSQL 是 Not Only SQL 的缩写,意即"不仅仅是SQL"的意思,泛指非关系型的数据库。强调Key-Value Stores和文档数据库的优点。 NoSQL产品是传统关系型数据库的功能阉割版本,通过减少用不到或…...

分支和循环
通常来说,C语言是结构化的程序设计语言,这里的结构包括顺序结构、选择结构、循环结构,C语言能够实现这三种结构,如果我们仔细分析,我们日常生活中所见的事情都可以拆分为这三种结构或者它们的组合。 下面我会仔细讲解我…...
MyBatis-xml版本
MyBatis 是一款优秀的持久层框架 MyBatis中文网https://mybatis.net.cn/ 添加依赖 <dependencies><!--mysql驱动--><dependency><groupId>mysql</groupId><artifactId>mysql-connector-java</artifactId><version>5.1.47<…...

在eclipse中安装python插件:PyDev
在eclipse中安装插件PyDev,就可以在eclipse中开发python了。 PyDev的官网:https://www.pydev.org/ 不过可以直接在eclipse中用Marketplace安装(备注:有可能一次安装不成功,是因为下载太慢了,多试几次&…...

25、pytest的测试报告插件allure
allure简介 在这里,你将找到使用allure创建、定制和理解测试报告所需的一切。开始让你的测试沟通更清晰,更有影响力。 Allure Report是一个实用程序,它处理由兼容的测试框架收集的测试结果并生成HTML报告。 安装allure 1、确保安装了Java…...

从零开始学习 JavaScript APl(七):实例解析关于京东案例头部案例和放大镜效果!
大家好关于JS APl 知识点已经全部总结了,第七部部分全部都是案例部分呢!!(素材的可以去百度网盘去下载!!!) 目录 前言 一、个人实战文档 放大镜效果 思路分析: 关于其它…...

使用Pytoch实现Opencv warpAffine方法
随着深度学习的不断发展,GPU/NPU的算力也越来越强,对于一些传统CV计算也希望能够直接在GPU/NPU上进行,例如Opencv的warpAffine方法。Opencv的warpAffine的功能主要是做仿射变换,如果不了解仿射变换的请自行了解。由于Pytorch的图像…...

Hello World
世界上最著名的程序 from fastapi import FastAPIapp FastAPI()app.get("/") async def root():return {"message": "Hello World"}app.get("/hello/{name}") async def say_hello(name: str):return {"message": f"…...

【Python】Python读Excel文件生成xml文件
目录 前言 正文 1.Python基础学习 2.Python读取Excel表格 2.1安装xlrd模块 2.2使用介绍 2.2.1常用单元格中的数据类型 2.2.2 导入模块 2.2.3打开Excel文件读取数据 2.2.4常用函数 2.2.5代码测试 2.2.6 Python操作Excel官方网址 3.Python创建xml文件 3.1 xml语法…...

c++--类型行为控制
1.c的类 1.1.c的类关键点 c类型的关键点在于类存在继承。在此基础上,类存在构造,赋值,析构三类通用的关键行为。 类型提供了构造函数,赋值运算符,析构函数来让我们控制三类通用行为的具体表现。 为了清楚的说明类的构…...

笔记64:Bahdanau 注意力
本地笔记地址:D:\work_file\(4)DeepLearning_Learning\03_个人笔记\3.循环神经网络\第10章:动手学深度学习~注意力机制 a a a a a a a a a a a...
面试官问:如何手动触发垃圾回收?幸好昨天复习到了
在Java中,手动触发垃圾回收可以使用 System.gc() 方法。但需要注意,调用 System.gc() 并不能确保立即执行垃圾回收,因为具体的垃圾回收行为是由Java虚拟机决定的,而不受程序员直接控制。 public class GarbageCollectionExample …...

操作系统的运行机制+中断和异常
一、CPU状态 在CPU设计和生产的时候就划分了特权指令和非特叔指令,因此CPU执行一条指令前就能断出其类型 CPU有两种状态,“内核态”和“用户态” 处于内核态时,说明此时正在运行的是内核程序,此时可以执行特权指令。 处于用户态…...

Python实战:批量加密Excel文件指南
更多Python学习内容:ipengtao.com 大家好,我是彭涛,今天为大家分享 Python实战:批量加密Excel文件指南,全文3800字,阅读大约10分钟。 在日常工作中,保护敏感数据是至关重要的。本文将引导你通过…...

【机器视觉】单目测距——运动结构恢复
ps:图是随便找的,为了凑个封面 前言 在前面对光流法进行进一步改进,希望将2D光流推广至3D场景流时,发现2D转3D过程中存在尺度歧义问题,需要补全摄像头拍摄图像中缺失的深度信息,否则解空间不收敛…...

Mac软件卸载指南,简单易懂!
刚和Adobe分手,它却总在Library里给你写"回忆录"?卸载的Final Cut Pro像电子幽灵般阴魂不散?总是会有残留文件,别慌!这份Mac软件卸载指南,将用最硬核的方式教你"数字分手术"࿰…...

安宝特方案丨船舶智造的“AR+AI+作业标准化管理解决方案”(装配)
船舶制造装配管理现状:装配工作依赖人工经验,装配工人凭借长期实践积累的操作技巧完成零部件组装。企业通常制定了装配作业指导书,但在实际执行中,工人对指导书的理解和遵循程度参差不齐。 船舶装配过程中的挑战与需求 挑战 (1…...

推荐 github 项目:GeminiImageApp(图片生成方向,可以做一定的素材)
推荐 github 项目:GeminiImageApp(图片生成方向,可以做一定的素材) 这个项目能干嘛? 使用 gemini 2.0 的 api 和 google 其他的 api 来做衍生处理 简化和优化了文生图和图生图的行为(我的最主要) 并且有一些目标检测和切割(我用不到) 视频和 imagefx 因为没 a…...

安全突围:重塑内生安全体系:齐向东在2025年BCS大会的演讲
文章目录 前言第一部分:体系力量是突围之钥第一重困境是体系思想落地不畅。第二重困境是大小体系融合瓶颈。第三重困境是“小体系”运营梗阻。 第二部分:体系矛盾是突围之障一是数据孤岛的障碍。二是投入不足的障碍。三是新旧兼容难的障碍。 第三部分&am…...

PHP 8.5 即将发布:管道操作符、强力调试
前不久,PHP宣布了即将在 2025 年 11 月 20 日 正式发布的 PHP 8.5!作为 PHP 语言的又一次重要迭代,PHP 8.5 承诺带来一系列旨在提升代码可读性、健壮性以及开发者效率的改进。而更令人兴奋的是,借助强大的本地开发环境 ServBay&am…...

从物理机到云原生:全面解析计算虚拟化技术的演进与应用
前言:我的虚拟化技术探索之旅 我最早接触"虚拟机"的概念是从Java开始的——JVM(Java Virtual Machine)让"一次编写,到处运行"成为可能。这个软件层面的虚拟化让我着迷,但直到后来接触VMware和Doc…...
ThreadLocal 源码
ThreadLocal 源码 此类提供线程局部变量。这些变量不同于它们的普通对应物,因为每个访问一个线程局部变量的线程(通过其 get 或 set 方法)都有自己独立初始化的变量副本。ThreadLocal 实例通常是类中的私有静态字段,这些类希望将…...
Django RBAC项目后端实战 - 03 DRF权限控制实现
项目背景 在上一篇文章中,我们完成了JWT认证系统的集成。本篇文章将实现基于Redis的RBAC权限控制系统,为系统提供细粒度的权限控制。 开发目标 实现基于Redis的权限缓存机制开发DRF权限控制类实现权限管理API配置权限白名单 前置配置 在开始开发权限…...

【PX4飞控】mavros gps相关话题分析,经纬度海拔获取方法,卫星数锁定状态获取方法
使用 ROS1-Noetic 和 mavros v1.20.1, 携带经纬度海拔的话题主要有三个: /mavros/global_position/raw/fix/mavros/gpsstatus/gps1/raw/mavros/global_position/global 查看 mavros 源码,来分析他们的发布过程。发现前两个话题都对应了同一…...