详解Qt中访问数据库
在Qt中访问数据库涉及到几个关键步骤,主要包括加载数据库驱动、建立数据库连接、执行SQL语句、读取结果等。下面将详细介绍这些步骤,并给出一个简单的示例,这里假设使用的是SQLite数据库。
记得首先在pro文件中添加QT += sql
1. 加载数据库驱动
Qt通过数据库驱动程序来支持不同类型的数据库,例如SQLite、MySQL、PostgreSQL等。在实际使用前,通常需要确保已经包含了相应的数据库驱动模块。对于SQLite,由于Qt内置了对其的支持,无需额外安装驱动。
#include <QSqlDatabase>
#include <QSqlQuery>
#include <QSqlError>// 如果使用其他数据库,可能需要包含对应的头文件,例如:
// #include <QMYSQLDriver> // 对于MySQL
// #include <QPSQLDriver> // 对于PostgreSQL
注意: 当前Qt是否已经安装将要使用的数据库驱动,可以使用QSqlDatabase::drivers()
去查看
#include <QtSql>
#include <QDebug> int main(int argc, char *argv[]) { QCoreApplication a(argc, argv); //输出所有已经支持的数据库驱动QStringList drivers = QSqlDatabase::drivers(); foreach (const QString &driver, drivers) { qDebug() << driver; } return a.exec();
}
2. 创建数据库连接
创建数据库连接通常涉及指定数据库类型(如果是SQLite,则通常不需要用户名、密码和数据库地址,因为SQLite数据库文件是本地文件)。
QSqlDatabase db = QSqlDatabase::addDatabase("QSQLITE"); // 对于SQLite
// 或者
// QSqlDatabase db = QSqlDatabase::addDatabase("QMYSQL"); // 对于MySQL
// QSqlDatabase db = QSqlDatabase::addDatabase("QPSQL"); // 对于PostgreSQLdb.setHostName("localhost"); // 对于远程数据库,设置主机名或IP
db.setDatabaseName("/path/to/mydatabase.db"); // 对于SQLite,这是数据库文件路径;对于其他数据库则是数据库名
db.setUserName("username"); // 非SQLite数据库的用户名
db.setPassword("password"); // 非SQLite数据库的密码if (!db.open()) {qDebug() << "Failed to connect to database: " << db.lastError().text();return false; // 连接失败
}
3. 执行SQL查询或命令
使用QSqlQuery
类执行SQL语句。
QSqlQuery query;// 插入数据
query.prepare("INSERT INTO MyTable (column1, column2) VALUES (?, ?)");
query.addBindValue(value1);
query.addBindValue(value2);
if (!query.exec()) {qDebug() << "Insert error: " << query.lastError().text();
} else {qDebug() << "Row inserted successfully.";
}// 查询数据
query.clear();
query.prepare("SELECT * FROM MyTable WHERE id = ?");
query.bindValue(0, someId);
if (query.exec()) {while (query.next()) {QString value1 = query.value(0).toString();int value2 = query.value(1).toInt();// ... 处理查询结果 ...}
} else {qDebug() << "Select error: " << query.lastError().text();
}
4. 关闭数据库连接
在完成所有数据库操作后,关闭连接以释放资源。
db.close();
示例完整代码片段
#include <QCoreApplication>
#include <QSqlDatabase>
#include <QSqlQuery>
#include <QSqlError>int main(int argc, char *argv[])
{QCoreApplication a(argc, argv);QSqlDatabase db = QSqlDatabase::addDatabase("QSQLITE");db.setDatabaseName(":memory:"); // 使用内存数据库,也可以替换为实际文件路径if (!db.open()) {qDebug() << "Cannot open database: " << db.lastError().text();return 1;}// 创建表QSqlQuery createQuery;createQuery.exec("CREATE TABLE IF NOT EXISTS MyTable (id INTEGER PRIMARY KEY, column1 VARCHAR(40), column2 INT)");// 插入数据QSqlQuery insertQuery;insertQuery.prepare("INSERT INTO MyTable (column1, column2) VALUES (?, ?)");insertQuery.addBindValue("Example Value");insertQuery.addBindValue(123);if (!insertQuery.exec()) {qDebug() << "Insert error: " << insertQuery.lastError().text();} else {qDebug() << "Row inserted successfully.";}// 查询数据QSqlQuery selectQuery;selectQuery.prepare("SELECT * FROM MyTable");if (selectQuery.exec()) {while (selectQuery.next()) {QString column1Value = selectQuery.value(1).toString();int column2Value = selectQuery.value(2).toInt();qDebug() << "Column1: " << column1Value << ", Column2: " << column2Value;}} else {qDebug() << "Select error: " << selectQuery.lastError().text();}db.close();return a.exec();
}
以上代码演示了如何使用Qt连接SQLite数据库,执行创建表、插入数据和查询数据的操作。
注意
- 对于不同的数据库类型,配置连接参数的方式会有所不同,但执行SQL的基本模式是相似的。
- 上述代码中使用的Sqlite数据库的内存模式。访问速度非常快。适合作为临时的缓存数据库使用。
打开多个数据库
在Qt程序中同时打开和操作多个数据库,可以通过创建多个QSqlDatabase
实例来实现。每个数据库实例都有自己的名字(connection name),这样可以区分不同数据库连接。以下是一个简单的示例,展示如何同时打开两个SQLite数据库:
#include <QApplication>
#include <QSqlDatabase>
#include <QSqlQuery>
#include <QDebug>int main(int argc, char *argv[])
{QApplication a(argc, argv);// 打开第一个SQLite数据库QSqlDatabase db1 = QSqlDatabase::addDatabase("QSQLITE");db1.setDatabaseName("first_db.sqlite");if (!db1.open()) {qDebug() << "Failed to open the first database: " << db1.lastError().text();return 1;}// 执行第一个数据库的查询QSqlQuery query1(db1);query1.exec("CREATE TABLE IF NOT EXISTS Table1 (ID INTEGER PRIMARY KEY, Name TEXT)");if (!query1.isActive())qDebug() << "Error creating table in first DB: " << query1.lastError().text();// 打开第二个SQLite数据库QSqlDatabase db2 = QSqlDatabase::addDatabase("QSQLITE");db2.setDatabaseName("second_db.sqlite");if (!db2.open()) {qDebug() << "Failed to open the second database: " << db2.lastError().text();db1.close();return 1;}// 执行第二个数据库的查询QSqlQuery query2(db2);query2.exec("CREATE TABLE IF NOT EXISTS Table2 (ID INTEGER PRIMARY KEY, Name TEXT)");if (!query2.isActive())qDebug() << "Error creating table in second DB: " << query2.lastError().text();// 不论操作成功与否,记得在不再使用时关闭数据库连接db1.close();db2.close();return a.exec();
}
在上述示例中,我们创建了两个不同的SQLite数据库连接,并分别为它们创建了表。当然,你可以根据需要连接不同类型(如MySQL、PostgreSQL等)的数据库,只需在调用addDatabase
时指定正确的数据库驱动名称即可。
务必注意,当在多线程环境中操作数据库时,即使Qt的数据库模块是线程安全的,也应该确保在同一时刻仅在一个线程中操作单个数据库连接,或者正确地管理线程间的同步,防止数据竞争问题。同时,也要确保在不再需要时关闭数据库连接,以避免资源泄露。
使用建议
- 数据库处理通常在主线程,如果必要在其他线程中访问数据库,切记在哪个线程访问数据库就在哪个线程中打开数据库。
- 如果甲方对数据库没有硬性要求,建议使用sqlite。
- 在连接远程数据库进行查询数据的时候,如果遇到访问缓慢情况,建议开启
QSqlQuery
的setForwardOnly(true)
。可以看到奇迹的发生。
相关文章:
详解Qt中访问数据库
在Qt中访问数据库涉及到几个关键步骤,主要包括加载数据库驱动、建立数据库连接、执行SQL语句、读取结果等。下面将详细介绍这些步骤,并给出一个简单的示例,这里假设使用的是SQLite数据库。 记得首先在pro文件中添加QT sql 1. 加载数据库驱动…...

《QT实用小工具·三》偏3D风格的异型窗体
1、概述 源码放在文章末尾 可以在窗体中点击鼠标左键进行图片切换,项目提供了一些图片素材,整体风格偏向于3D类型,也可以根据需求自己放置不同的图片。 下面是demo演示: 项目部分代码如下所示: 头文件部分ÿ…...

如何优化TCP?TCP的可靠传输机制是什么?
在网络世界中,传输层协议扮演着至关重要的角色,特别是TCP协议,以其可靠的数据传输特性而广受青睐。然而,随着网络的发展和数据量的激增,传统的TCP协议在效率方面遭遇了挑战。小编将深入分析TCP的可靠性传输机制&#x…...

DFS(基础,回溯,剪枝,记忆化)搜索
DFS基础 DFS(深度优先搜索) 基于递归求解问题,而针对搜索的过程 对于问题的介入状态叫初始状态,要求的状态叫目标状态 这里的搜索就是对实时产生的状态进行分析检测,直到得到一个目标状态或符合要求的最佳状态为止。对于实时产生新的状态…...

基于Scala开发Spark ML的ALS推荐模型实战
推荐系统,广泛应用到电商,营销行业。本文通过Scala,开发Spark ML的ALS算法训练推荐模型,用于电影评分预测推荐。 算法简介 ALS算法是Spark ML中实现协同过滤的矩阵分解方法。 ALS,即交替最小二乘法(Alte…...
Go语言和Java编程语言的主要区别
目录 1.设计理念: 2.语法: 3.性能: 4.并发性: 5.内存管理: 6.标准库: 7.社区和支持: 8.应用领域: Go(也称为Golang)和Java是两种不同的编程语言&…...
【TypeScript系列】与其它构建工具整合
与其它构建工具整合 构建工具 BabelBrowserifyDuoGruntGulpJspmWebpackMSBuildNuGet Babel 安装 npm install babel/cli babel/core babel/preset-typescript --save-dev.babelrc {"presets": ["babel/preset-typescript"] }使用命令行工具 ./node_…...

Java | Leetcode Java题解之第12题整数转罗马数字
题解: 题解: class Solution {String[] thousands {"", "M", "MM", "MMM"};String[] hundreds {"", "C", "CC", "CCC", "CD", "D", "DC…...

哈佛大学商业评论 --- 第五篇:智能眼镜之战
AR将全面融入公司发展战略! AR将成为人类和机器之间的新接口! AR将成为人类的关键技术之一! 请将此文转发给您的老板! --- 专题作者:Michael E.Porter和James E.Heppelmann 虽然物理世界是三维的,但大多…...

paddlepaddle模型转换onnx指导文档
一、检查本机cuda版本 1、右键找到invdia控制面板 2、找到系统信息 3、点开“组件”选项卡, 可以看到cuda版本,我们这里是cuda11.7 cuda驱动版本为516.94 二、安装paddlepaddle环境 1、获取pip安装命令 ,我们到paddlepaddle官网ÿ…...

图像处理与视觉感知---期末复习重点(6)
文章目录 一、图像分割二、间断检测2.1 概述2.2 点检测2.3 线检测2.4 边缘检测 三、边缘连接3.1 概述3.2 Hough变换3.3 例子3.4 Hough变换的具体步骤3.5 Hough变换的法线表示形式3.6 Hough变换的扩展 四、阈值处理4.1 概述4.2 计算基本全局阈值算法4.3 自适应阈值 五、基于区域…...
git 如何删除本地和远程分支
删除本地分支 确认当前分支:首先,确保你没有在要删除的分支上。你可以通过运行git branch命令来查看当前的分支。 切换分支:如果你在要删除的分支上,需要先切换到另一个分支。例如,切换到main分支,可以使用…...
Kong基于QPS、IP限流
Rate Limiting限流插件 https://docs.konghq.com/hub/kong-inc/rate-limiting/ 它可以针对consumer ,credential ,ip ,service,path,header 等多种维度来进行限流.流量控制的精准度也有多种方式可以参考,比如可以做到秒级,分钟级,小时级等限流控制. 基于IP限流 源码地址&…...

基于springboot实现甘肃非物质文化网站系统项目【项目源码+论文说明】
摘要 现代经济快节奏发展以及不断完善升级的信息化技术,让传统数据信息的管理升级为软件存储,归纳,集中处理数据信息的管理方式。本甘肃非物质文化网站就是在这样的大环境下诞生,其可以帮助管理者在短时间内处理完毕庞大的数据信…...

【瑞萨RA6M3】1. 基于 vscode 搭建开发环境
基于 vscode 搭建开发环境 1. 准备2. 安装2.1. 安装瑞萨软件包2.2. 安装编译器2.3. 安装 cmake2.4. 安装 openocd2.5. 安装 ninja2.6. 安装 make 3. 生成初始代码4. 修改 cmake 脚本5. 调试准备6. 仿真 1. 准备 需要瑞萨仓库中的两个软件: MDK_Device_Packs.zipse…...

使用pip install替代conda install将packet下载到anaconda虚拟环境
问题描述 使用conda install 下载 stable_baseline3出现问题 一番搜索下是Anaconda.org缺少源 解决方法 首先使用管理员权限打开 anaconda prompt 然后激活目标环境:conda activate env_name 接着使用:conda env list查看目标env的位置 如D:\anacon…...

【HTML】常用CSS属性
文章目录 前言1、字体和文本属性2、边距和填充3、border边框4、列表属性 前言 上一篇我们学习了CSS扩展选择器以及它的继承性,对于页面元素样式设置相信大家都不陌生了。 这一篇我们就来看看具体都有哪些样式可以设置?又该如何设置? 喜欢的【…...
python中的print(f‘‘)具体用法
在Python中,print(f) 是格式化字符串(f-string)的语法,它允许你在字符串中嵌入表达式,这些表达式在运行时会被其值所替换。f 或 F 前缀表示这是一个格式化字符串字面量。 在 f 或 F 中的大括号 {} 内,你可…...
《青少年成长管理2024》022 “成长七要素之三:文化”4/5
《青少年成长管理2024》022 “成长七要素之三:文化”4/5 七、物质文化(一)什么是物质文化(二)物质文化的分类(三)人类物质文化最新成果有哪些(四)青少年了解物质文化的途…...

Linux(05) Debian 系统修改主机名
查看主机名 方法1:hostname hostname 方法2:cat etc/hostname cat /etc/hostname 如果在创建Linux系统的时候忘记修改主机名,可以采用以下的方式来修改主机名称。 修改主机名 注意,在linux中下划线“_”可能是无效的字符&…...
利用ngx_stream_return_module构建简易 TCP/UDP 响应网关
一、模块概述 ngx_stream_return_module 提供了一个极简的指令: return <value>;在收到客户端连接后,立即将 <value> 写回并关闭连接。<value> 支持内嵌文本和内置变量(如 $time_iso8601、$remote_addr 等)&a…...
【android bluetooth 框架分析 04】【bt-framework 层详解 1】【BluetoothProperties介绍】
1. BluetoothProperties介绍 libsysprop/srcs/android/sysprop/BluetoothProperties.sysprop BluetoothProperties.sysprop 是 Android AOSP 中的一种 系统属性定义文件(System Property Definition File),用于声明和管理 Bluetooth 模块相…...

20个超级好用的 CSS 动画库
分享 20 个最佳 CSS 动画库。 它们中的大多数将生成纯 CSS 代码,而不需要任何外部库。 1.Animate.css 一个开箱即用型的跨浏览器动画库,可供你在项目中使用。 2.Magic Animations CSS3 一组简单的动画,可以包含在你的网页或应用项目中。 3.An…...
Python+ZeroMQ实战:智能车辆状态监控与模拟模式自动切换
目录 关键点 技术实现1 技术实现2 摘要: 本文将介绍如何利用Python和ZeroMQ消息队列构建一个智能车辆状态监控系统。系统能够根据时间策略自动切换驾驶模式(自动驾驶、人工驾驶、远程驾驶、主动安全),并通过实时消息推送更新车…...

从 GreenPlum 到镜舟数据库:杭银消费金融湖仓一体转型实践
作者:吴岐诗,杭银消费金融大数据应用开发工程师 本文整理自杭银消费金融大数据应用开发工程师在StarRocks Summit Asia 2024的分享 引言:融合数据湖与数仓的创新之路 在数字金融时代,数据已成为金融机构的核心竞争力。杭银消费金…...

springboot 日志类切面,接口成功记录日志,失败不记录
springboot 日志类切面,接口成功记录日志,失败不记录 自定义一个注解方法 import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target;/***…...
规则与人性的天平——由高考迟到事件引发的思考
当那位身着校服的考生在考场关闭1分钟后狂奔而至,他涨红的脸上写满绝望。铁门内秒针划过的弧度,成为改变人生的残酷抛物线。家长声嘶力竭的哀求与考务人员机械的"这是规定",构成当代中国教育最尖锐的隐喻。 一、刚性规则的必要性 …...

智警杯备赛--excel模块
数据透视与图表制作 创建步骤 创建 1.在Excel的插入或者数据标签页下找到数据透视表的按钮 2.将数据放进“请选择单元格区域“中,点击确定 这是最终结果,但是由于环境启不了,这里用的是自己的excel,真实的环境中的excel根据实训…...

生信服务器 | 做生信为什么推荐使用Linux服务器?
原文链接:生信服务器 | 做生信为什么推荐使用Linux服务器? 一、 做生信为什么推荐使用服务器? 大家好,我是小杜。在做生信分析的同学,或是将接触学习生信分析的同学,<font style"color:rgb(53, 1…...
JS的传统写法 vs 简写形式
一、条件判断与逻辑操作 三元运算符简化条件判断 // 传统写法 let result; if (someCondition) {result yes; } else {result no; }// 简写方式 const result someCondition ? yes : no;短路求值 // 传统写法 if (condition) {doSomething(); }// 简写方式 condition &…...