QSqlQuery简单使用
- 1 执行一个查询
- 2 浏览查询结果
- 3 插入、更新和删除记录
- 关键代码
- Qt 程序获取程序所在路径、用户目录路径、临时文件夹等特殊路径的方法
- 程序所在路径
- 当前工作目录
- 用户目录路径
- 我的文档路径
- 桌面路径
- 程序数据存放路径
- 临时文件路径
- Qt中路径问题小结
原文链接:https://blog.csdn.net/Charmve/article/details/109401105
QSqlQuery类提供了一个用于执行SQL语句和浏览查询的结果集的接口。
QSqlQueryModel和QSqlTableModel类提供了一个用于访问数据库的高级接口,这将在下一节介绍。如果你不熟悉SQL,你可以直接使用下一节介绍的高级接口类。
1 执行一个查询
要执行一个SQL语句,简单的创建一个QSqlQuery对象,然后调用QSqlQuery::exec()方法即可,如下所示:
QSqlQuery query;
query.exec("SELECT name, salary FROM employee WHERE salary > 50000");
QSqlQuery的构造函数接受一个可选的QSqlDatabase对象参数,用来指定将使用哪一个数据库连接。在上面的例子中,我们并没有指定任何连接,因此使用默认连接。如果执行查询时出错,exec()将返回false,错误原因可以通过QSqlQuery::lastError()查看。
2 浏览查询结果
QSqlQuery提供了一个访问一条查询结果记录的方法。在调用了exec()方法之后,QSqlQuery的内部指针定位到了第一条记录之前的位置。我们必须调用一次QSqlQuery::next()内部指针就移动到第一条记录上,然后重复调用next()方法就可以移动到其他记录上,直到该函数返回false为止。下面给出一个典型的循环依次遍历所有的记录:
while (query.next()) {QString name = query.value(0).toString();int salary = query.value(1).toInt();qDebug() << name << salary;
}
QSqlQuery::value()函数返回当前记录的一个字段值。字段的索引号从0开始计数。QSqlQuery::value()返回一个QVariant,这个类型可以保存各种C++和核心的Qt数据类型,比如int, QString和QByteArray等。不同的数据库类型将自动地映射到最近的Qt等价类型。在上面的代码中,我们使用了QVariant::toString()和QVariant::toInt()将变量类型转换为了QString和int。
补充: 关于value()方法的参数-索引号的取值范围问题,我们只知道第一个字段对应的索引号为0,但是并不知道最后一个字段的索引号的时候,我们可以遍历所有可能值,由于当value()函数的索引号非法时,会返回一个非法的QVariant变量,因此,我们可以通过判断返回的QVariant变量的合法性来得知是否已经到了最后一个字段。下面给出一段示例代码:
while (query.next()){int i = 0;QString result;QVariant temp;while ((temp = query.value(i++)).isValid()){result += temp.toString();result += " | ";}ui->textBrowser->append(result);
}
下面给出一个MYSQL与Qt支持的数据类型之间的转换关系表:
SQLITE3与Qt支持的数据类型之间的转换关系表如下:
你可以使用QSqlQuery::next(),QSqlQuery::previous(), QSqlQuery::first()和QSqlQuery::last()来前后移动指向记录位置的指针。可以使用QSqlQuery::at()返回当前行的索引号,如果对应的数据库支持,可以使用QSqlQuery::size()返回查询结果集的总的行数(MYSQL是支持的)。
要验证一个数据库驱动是否支持某一个特性,QSqlDriver::hasFeature()。下面的示例我们调用调用QSqlQuery::size()来检查底层的数据库是否支持此特性,否则,我们只能浏览到最后一条记录,利用查询的位置来得到查询结果中一共有多少条记录。
QSqlQuery query;
int numRows;
query.exec("SELECT name, salary FROM employee WHERE salary > 50000");QSqlDatabase defaultDB = QSqlDatabase::database();
if (defaultDB.driver()->hasFeature(QSqlDriver::QuerySize)) {numRows = query.size();
} else {// this can be very slowquery.last();numRows = query.at() + 1;
}
如果你在遍历查询结果集的时候只使用正值参数的next()和seek(),你可以在调用exec()之前调用QSqlQuery::setForwartOnly(true)。当你在操作大量结果集的时候,这是一个简单的优化单可以明显加速你的查询。
3 插入、更新和删除记录
QSqlQuery可以执行任意的SQL语句。下面的示例展示了使用INSERT语句向表中插入一条记录:
QSqlQuery query;
query.exec("INSERT INTO employee (id, name, salary) ""VALUES (1001, 'Thad Beaumont', 65000)");
如果你想要同时插入许多条记录,一个有效的方法就是将查询语句和真实的值分离,这个可以使用占位符来实现。Qt支持两种占位符:名称绑定和位置绑定。下面是一个名称绑定的示例:
QSqlQuery query;
query.prepare("INSERT INTO employee (id, name, salary) ""VALUES (:id, :name, :salary)");
query.bindValue(":id", 1001);
query.bindValue(":name", "Thad Beaumont");
query.bindValue(":salary", 65000);
query.exec();
下面的代码时位置绑定的示例:
QSqlQuery query;
query.prepare("INSERT INTO employee (id, name, salary) ""VALUES (?, ?, ?)");
query.addBindValue(1001);
query.addBindValue("Thad Beaumont");
query.addBindValue(65000);
query.exec();
这两种语句对于Qt提供的所有数据库驱动都支持。当需要插入多条记录时,只需要调用 一次QSqlQuery::prepare(),然后使用多次bindValue()或者addBindValue()来绑定需要的数据,最后再调用一次exec()函数就可以了。
抛开性能,使用占位符的一个好处就是你可以指定任意的值而不用担心缺少特殊字符。
更新一条记录与插入操作非常相似:
QSqlQuery query;
query.exec("UPDATE employee SET salary = 70000 WHERE id = 1003");
你也可以使用名称或位置绑定来关联实际的参数值。
最后,下面给出一条删除语句:
QSqlQuery query;
query.exec("DELETE FROM employee WHERE id = 1007");
4 事务
如果底层的数据库引擎支持事务,那么QSqlDriver::hasFeature(QSqlDriver::Transactions)会返回true。可以使用QSqlDatabase::transaction()启动一个事务,然后编写一些希望在事务中执行的SQL语句,最后调用QSqlDatabase::commit()或者QSqlDatabase::rollback()。当使用事务时必须在创建查询以前就开始事务,例如:
QSqlDatabase::database().transaction();
QSqlQuery query;
query.exec("SELECT id FROM employee WHERE name = 'Torild Halvorsen'");
if (query.next()) {int employeeId = query.value(0).toInt();query.exec("INSERT INTO project (id, name, ownerid) ""VALUES (201, 'Manhattan Project', "+ QString::number(employeeId) + ')');
}
QSqlDatabase::database().commit();
事务可以保证一个复杂操作的原子性,就是对于一个数据库操作序列,这些操作要么全部做完,要么一条也不做,它是一个不可分割的工作单位。
关键代码
对于数据库的操作:打开数据库
_db = QSqlDatabase::addDatabase("QSQLITE");_db.setDatabaseName("demo.db");// 数据库文件是否在应用程序目录下,因为在编码时,应用程序默认路径在.pro文件夹下,但是实际exe路径在的其debug或者release文件夹下// 补充:不论linux还是windows,统一使用单斜杠作为分隔,在linux和windows下都能识别,能更好的跨平台
#if 0// 发布时QString path = qApp->applicationDirPath()+"/"+"demo.db";
#else// 编译时QString path = qApp->applicationDirPath()+"/../"+"demo.db";
#endifif(QFile::exists(path)){// 存在则打开_db.open();}else{// 不存在打开并创建数据库表 补充:SQLite是基于文件的数据库,当打开时如果数据库文件不存在将自动创建一个_db.open();QSqlQuery query;// set为sqlite关键字,不能使用bool bRet = query.exec("create table init(" \"no INT PRIMARY KEY NOT NULL,"\"name TEXT NOT NULL,"\"content TEXT,"\"description TEXT"\");");if(!bRet) {_db.close();QFile::remove(path);}else{// 此处演示了三种不同insert插入方法query.exec("insert into init values" \"(1, '启动界面背景', 'images/background.jpg', '请使用1920*1080的图片,图片格式可以为png、jpg。');");query.exec("insert into init " \"(no,name,content,description) values(2, '欢迎视频', '','进入启动界面后,循环播放的视频,必须为mp4格式,如果为空则循环播放欢迎音乐');");query.exec("insert into init " \"(no,name,description) values (3, '欢迎音乐','进去启动界面后,循环播放的音频文件,可选择mp3,wav格式');");}}
Qt 程序获取程序所在路径、用户目录路径、临时文件夹等特殊路径的方法
原文链接:https://blog.csdn.net/liyuanbhu/article/details/53710249
Qt 程序获取程序所在路径、用户目录路径、临时文件夹等特殊路径的方法
经常我们的程序中需要访问一些特殊的路径,比如程序所在的路径、用户目录路径、临时文件夹等。在 Qt 中实现这几个功能所用的方法虽然都不难,但是各不相同,每次用到时还要现去查,很不方便。因此就写了这篇博客,把这几种需求的实现方式总结了一下。算是个备忘录吧。
程序所在路径
获取程序所在路径,QCoreApplication 类里就实现了相关的功能:
QString QCoreApplication::applicationDirPath()
比如我们有一个程序在:
C:/Qt/examples/tools/regexp/regexp.exe
那么 qApp->applicationDirPath() 的结果是:
C:/Qt/examples/tools/regexp
如果除了程序所在路径,我们还想要程序的完整名称。那么可以这么写:
qApp->applicationFilePath()
还是上面的例子,结果是:
C:/Qt/examples/tools/regexp/regexp.exe
当前工作目录
QDir 提供了一个静态函数 currentPath() 可以获取当前工作目录,函数原型如下:
QString QDir::currentPath()
如果我们是双击一个程序运行的,那么程序的工作目录就是程序所在目录。
如果是在命令行下运行一个程序,那么运行程序时在命令行的哪个目录,那个目录就是当前目录。
用户目录路径
Qt 4 中的方法。下面的方法只对 Qt 4 有效,Qt 5 已经删除了 storageLocation() 方法。
QDesktopServices::storageLocation(QDesktopServices::HomeLocation);
Qt 5 中引入的方法。
QStandardPaths::writableLocation(QStandardPaths::HomeLocation);
或者
QStandardPaths::standardLocations(QStandardPaths::HomeLocation);
这两个方法的区别是 standardLocations() 返回值是 QStringList。当然对于 HomeLocation 来说这个 QStringList 中只有一个 QString。
还有另外一种方法,利用 QDir 类的一个静态函数:
QDir::homePath();
我的文档路径
Qt 4 中的方法。下面的方法只对 Qt 4 有效,Qt 5 已经删除了 storageLocation() 方法。
QDesktopServices::storageLocation(QDesktopServices::DocumentsLocation);
Qt 5 中引入的方法。
QStandardPaths::writableLocation(QStandardPaths::DocumentsLocation);
QStandardPaths::standardLocations(QStandardPaths::DocumentsLocation);
桌面路径
Qt 4 中的方法。下面的方法只对 Qt 4 有效,Qt 5 已经删除了 storageLocation() 方法。
QDesktopServices::storageLocation(QDesktopServices::DesktopLocation);
Qt 5 中引入的方法。
QStandardPaths::writableLocation(QStandardPaths::DesktopLocation);
QStandardPaths::standardLocations(QStandardPaths::DesktopLocation);
程序数据存放路径
通常我们会将程序所需的一些数据存入注册表。但是有时需要存储的数据太多,放在注册表中就不适合了。这时我们就要找个专门的地方来放数据。以前我喜欢将数据直接放到程序所在目录,但是后来发现我的程序运行时经常没有权限对这个目录下的文件进行写操作。后来发现其实 Qt 早就替我们考虑过这些问题了。
Qt 4 中的方法。下面的方法只对 Qt 4 有效,Qt 5 已经删除了 storageLocation() 方法。
QDesktopServices::storageLocation(QDesktopServices::DataLocation);
Qt 5 中引入的方法。
QStandardPaths::writableLocation(QStandardPaths::AppDataLocation);
QStandardPaths::standardLocations(QStandardPaths::AppDataLocation);
Qt 5.5 中引入了另一种方法:
QStandardPaths::writableLocation(QStandardPaths::AppConfigLocation);
QStandardPaths::standardLocations(QStandardPaths::AppConfigLocation);
这个方法一般来说和上面的方法得到的结果是相同的。按照 Qt 帮助文档的解释,这个方法可以确保返回的路径非空。所以我认为应该优先选用这个方法。
临时文件路径
Qt 4 中的方法。下面的方法只对 Qt 4 有效,Qt 5 已经删除了 storageLocation() 方法。
QDesktopServices::storageLocation(QDesktopServices::TempLocation);
Qt 5 中引入的方法。
QStandardPaths::writableLocation(QStandardPaths::TempLocation);
QStandardPaths::standardLocations(QStandardPaths::TempLocation);
更传统的方法是利用 QDir 的一个静态函数 tempPath()。
QDir::tempPath();
在这个目录下生成临时文件和临时目录需要用到另外两个类: QTemporaryFile 和 QTemporaryDir。就不展开介绍了,大家可以参考 qt 的帮助文档。
至此,常用的各种特殊路径就介绍的差不多了。剩下还有些不常用的,可以参考 QStandardPaths 类的介绍。
Qt中路径问题小结
原文链接:https://blog.csdn.net/Andy_93/article/details/52831175
在做Qt项目的时候,我们难免遇到到文件路径问题。 如QFile file(“text.txt”)加载不成功、QPixmap(“…/text.png”) 加载图片不成功等等。今天就来做一个关于Qt路径问题的小结!
首先需要弄清楚两个概念:绝对路径与相对路径
绝对路径: 文件在硬盘上真正存在的路径。例如“text.jpg”这个图片是存放在硬盘
的“E:\book\目录下,那么 “text.jpg”这个图片的绝对路径就是“E:\book\text.jpg"
相对路径:由这个文件所在的路径引起的跟其它文件(或文件夹)的路径关系
如当前file1的路径是/user/File/file1/file1.txt,
File2的路径是/user/File/file2/file2.txt
那么file2相对于file的路径就是“…/file2/file2.txt”。
相对路径里常使用“…/”来表示上一级目录
弄清楚了上面两个概念,在来弄清楚我们常用到的“: /”与"./":
./ : 表示当前路径 如“./log/log1.txt” 表示当前路径下的log目录下的log1.txt
: / : 表示对资源的引用,引用资源文件路径 如“:/image/start.png” 表示资源文件里面定义的文件start.png
现在我们来解释下QFile file(“text.txt”)加载不成功 ,如果text.txt是在可执行文件同级目录下,是可以打开的
的,否则则打不开
下面讲几个Qt常用的获取路径的函数
1 .获取应用程序可执行文件所在的目录: QCoreApplication::applicationDirPath();QString applicationDirPath;applicationDirPath = QCoreApplication::applicationDirPath();qDebug()<<"applicationDirPath"<<applicationDirPath;2.获取应用程序可执行文件的文件路径: QCoreApplication::applicationFilePath();QString applicationFilePath;applicationFilePath = QCoreApplication::applicationFilePath();qDebug()<<"applicationFilePath"<<applicationFilePath;3.获取应用程序当前工作目录的绝对路径:QString QDir::currentPath() (这个类似于“./”操作)QString currentPath;QDir dir;currentPath=dir.currentPath(); qDebug()<<"path"<<currentPath;将相对路径转化为绝对路径:QDir temDir("../../image.png");QString filePath = temDir.absolutePath();
这样就获得了一个文件的绝对路径了。
再来讲讲Qt资源系统:Qt 资源系统是一个跨平台的资源机制,用于将程序运行时所需要的资源以二进制的形式存储于可执行文件内部。 如果你的程序需要加载特定的资源(图标、文本翻译等), 那么,将其放置在资源文件中,就再也不需要担心这些文件的丢失。也就是说,如果你将资源以资源文件形式存储,它是会编译到可执行文件内部。
所以再来讲讲Qt资源文件qrc文件的创建。
方法1、使用qtdesinger创建.qrc文件
(1)在资源管理器窗口中选择新建一个资源文件.qrc,保存文件即可;
(2)在资源管理器窗口中选择添加资源和文件即可;
方法2、手工写文件的方法,.qrc可以用记事本打开,大概的形式是这样的:
<RCC><qresource prefix="/" > //前缀<file>image/text.png</file></qresource></RCC>
然后在pro文件中添加一句
RESOURCES = xxx.qrc
这样在使用的时候QPixmap(“:/image/text.png”);就能加载成功了
同时为了以后我们修改text.png路径的时候我们还需要一处处的修改源代码,所以在建立资源qrc的时候,给每个资源文件起一个别名
如 :
<RCC><file alias="image_copy.png">images/copy.png</file></RCC>
这时该文件可以通过QPixmap(“:/image_copy.png”);能加载成功。
但通过QPixmap(“:/images/copy.png”);加载不成功,QT里“: /”表示对资源的引用,不是表示当前目录
==>使用别名是为了日后资源文件路径改变而去改动源代码。用别名后,只需要修改qrc文件即可!