Qt数据库(⽀持10种数据库)
Qt 提供了 QtSql 模块来提供平台独⽴的基于 SQL 的数据库操作。这⾥我们所说的“平台独⽴”,既包括操作系统平台,有包括各个数据库平台。另外,我们强调了“基于 SQL”,因为 NoSQL 数据库⾄今没有⼀个通⽤查询⽅法,所以不可能提供⼀种通⽤的 NoSQL 数据库的操作。Qt 的数据库操作还可以很⽅便的与 model/view 架构进⾏整合。通常来说,我们对数据库的操作更多地在于对数据库表的操作,⽽这正是 model/view 架构的长项。
Qt 使⽤QSqlDatabase表⽰⼀个数据库连接。更底层上,Qt 使⽤驱动(drivers)来与不同的数据库 API 进⾏交互。Qt 桌⾯版本提供了如下⼏种驱动:
驱动数据库
QDB2IBM DB2 (7.1 或更新版本)
QIBASE Borland InterBase
QMYSQL MySQL
QOCI Oracle Call Interface Driver
QODBC Open Database Connectivity (ODBC) – Microsoft SQL Server 及其它兼容 ODBC 的数据库
QPSQL PostgreSQL (7.3 或更新版本)
QSQLITE2SQLite 2
QSQLITE SQLite 3
QSYMSQL针对 Symbian 平台的SQLite 3
QTDS Sybase Adaptive Server (⾃ Qt 4.7 起废除)
不过,由于受到协议的限制,Qt 开源版本并没有提供上⾯所有驱动的⼆进制版本,⽽仅仅以源代码的形式提供。通常,Qt 只默认搭载 QSqlite 驱动(这个驱动实际还包括 Sqlite 数据库,也就是说,如果需要使⽤ Sqlite 的话,只需要该驱动即可)。我们可以选择把这些驱动作为 Qt 的⼀部分进⾏编译,也可以当作插件编译。
如果习惯于使⽤ SQL 语句,我们可以选择QSqlQuery类;如果只需要使⽤⾼层次的数据库接⼝(不关⼼ SQL 语法),我们可以选择QSqlTableModel和QSqlRelationalTableModel。我们只介绍QSqlQuery类的使⽤。
在使⽤时,我们可以通过
QSqlDatabase::drivers();
到系统中所有可⽤的数据库驱动的名字列表。我们只能使⽤出现在列表中的驱动。由于默认情况下,QtSql 是作为 Qt 的⼀个模块提供的。为了使⽤有关数据库的类,我们必须早 .pro ⽂件中添加这么⼀句:
QT += sql
这表⽰,我们的程序需要使⽤ Qt 的 core、gui 以及 sql 三个模块。注意,如果需要同时使⽤ Qt4 和 Qt5 编译程序,通常我们的 .pro ⽂件是这样的:
1 QT += core gui sql
2 greaterThan(QT_MAJOR_VERSION, 4): QT += widgets
这两句也很明确:Qt 需要加载 core、gui 和 sql 三个模块,如果主板本⼤于 4,则再添加 widgets 模块。
下⾯来看⼀个简单的函数:
1bool connect(const QString &dbName)
2 {
3 QSqlDatabase db = QSqlDatabase::addDatabase("QSQLITE");
4// db.setHostName("host");
5// db.setDatabaseName("dbname");
6// db.setUserName("username");
7// db.setPassword("password");
8 db.setDatabaseName(dbName);
9if (!db.open()) {
10 QMessageBox::critical(0, QObject::tr("Database Error"),
11 db.lastError().text());
12return false;
13 }
14return true;
15 }
我们使⽤connect()函数创建⼀个数据库连接。我们使⽤QSqlDatabase::addDatabase()静态函数完成这⼀请求,也就是创建了⼀个QSqlDatabase实例。注意,数据库连接使⽤⾃⼰的名字进⾏区分,⽽不是数据库的名字。例如,我们可以使⽤下⾯的语句:
QSqlDatabase db=QSqlDatabase::addDatabase("QSQLITE", QString("con%1").arg(dbName));
此时,我们是使⽤addDatabase()函数的第⼆个参数来给这个数据库连接⼀个名字。在这个例⼦中,⽤于区分这个数据库连接的名字是QString(“conn%1”).arg(dbName),⽽不是 “QSQLITE”。这个参数是可选的,如果不指定,系统会给出⼀个默认的名字QSqlDatabase::defaultConnection,此时,Qt 会创建⼀个默认的连接。如果你给出的名字与已存在的名字相同,新的连接会替换掉已有的连接。通过这种设计,我们可以为⼀个数据库建⽴多个连接。
我们这⾥使⽤的是 sqlite 数据库,只需要指定数据库名字即可。如果是数据库服务器,⽐如 MySQL,我们还需要指定主机名、端⼝号、⽤户名和密码,这些语句使⽤注释进⾏了简单的说明。
接下来我们调⽤了QSqlDatabase::open()函数,打开这个数据库连接。通过检查open()函数的返回值,我们可以判断数据库是不是正确打开。
QtSql 模块中的类⼤多具有lastError()函数,⽤于检查最新出现的错误。如果你发现数据库操作有任何问题,应该使⽤这个函数进⾏错误的检查。这⼀点我们也在上⾯的代码中进⾏了体现。当然,这只是最简单的实现,⼀般来说,更好的设计是,不要在数据库操作中混杂界⾯代码(并且将这个connect()函数放在⼀个专门的数据库操作类中)。
接下来我们可以在main()函数中使⽤这个connect()函数:
1int main(int argc, char *argv[])
2 {
3 QApplication a(argc, argv);
4if (connect("demo.db")) {
5 QSqlQuery query;
6if (!("CREATE TABLE student ("
7"id INTEGER PRIMARY KEY AUTOINCREMENT,"
8"name VARCHAR,"
9"age INT)")) {
10 QMessageBox::critical(0, QObject::tr("Database Error"),
11 query.lastError().text());
12return1;
13 }
批量更新sql语句14 } else {
15return1;
16 }
();
18 }
main()函数中,我们调⽤这个connect()函数打开数据库。如果打开成功,我们通过⼀个QSqlQuery实例执⾏了 SQL 语句,就是();。同样,我们使⽤其lastError()函数检查了执⾏结果是否正确。
注意这⾥的QSqlQuery实例的创建。我们并没有指定是为哪⼀个数据库连接创建查询对象,此时,系统会使⽤默认的连接,也就是使⽤没有第⼆个参数的addDatabase()函数创建的那个连接(其实就是名字为QSqlDatabase::defaultConnection 的默认连接)。如果没有这么⼀个连接,系统就会报错。也就是说,如果没有默认连接,我们在创建QSqlQuery对象时必须指明是哪⼀个QSqlDatabase对象,也就是addDatabase()的返回值。
我们还可以通过使⽤QSqlQuery::isActive()函数检查语句执⾏正确与否。如果QSqlQuery对象是活动的,该函数返回true。所谓“活动”,就是指该对象成功执⾏了exec()函数,但是还没有完成。如果需要设置为不活动的,可以使⽤finish()或者clear()函数,或者直接释放掉这个QSqlQuery对象。这⾥需要
注意的是,如果存在⼀个活动的 SELECT 语句,某些数据库系统不能成功完成connect()或者rollback()函数的调⽤。此时,我们必须⾸先将活动的 SELECT 语句设置成不活动的。
创建过数据库表 student 之后,我们开始插⼊数据,然后将其独取出来:
1if (connect("demo.db")) {
2 QSqlQuery query;
3 query.prepare("INSERT INTO student (name, age) VALUES (?, ?)");
4 QVariantList names;
5 names << "Tom" << "Jack" << "Jane" << "Jerry";
6 query.addBindValue(names);
7 QVariantList ages;
8 ages << 20 << 23 << 22 << 25;
9 query.addBindValue(ages);
10if (!Batch()) {
11 QMessageBox::critical(0, QObject::tr("Database Error"),
12 query.lastError().text());
13 }
14 query.finish();
15 ("SELECT name, age FROM student");
16while (()) {
17 QString name = query.value(0).toString();
18int age = query.value(1).toInt();
19 qDebug() << name << ": " << age;
20 }
21 } else {
22return1;
23 }
依旧连接到我们创建的 demo.db 数据库。我们需要插⼊多条数据,此时可以使⽤QSqlQuery::exec()函数⼀条⼀条插⼊数据,但是这⾥我们选择了另外⼀种⽅法:批量执⾏。⾸先,我们使⽤QSqlQuery::prepare()函数对这条 SQL 语句进⾏预处理,问号 ? 相当于占位符,预⽰着以后我们可以使⽤实际数据替换这些位置。简单说明⼀下,预处理是数据库提供的⼀种特性,它会将 SQL 语句进⾏编译,性能和安全性都要优于普通的 SQL 处理。在上⾯的代码中,我们使⽤⼀个字符串列表names 替换掉第⼀个问号的位置,⼀个整型列表 ages 替换掉第⼆个问号的位置,利⽤QSqlQuery::addBindValue()我们将实际数据绑定到这个预处理的 SQL 语句上。需要注意的是,names 和 ages 这两个列表⾥⾯的数据需要⼀⼀对应。然后我们调⽤QSqlQuery::execBatch()批量执⾏ SQL,之后结束该对象。这样,插⼊操作便完成了。
另外说明⼀点,我们这⾥使⽤了 ODBC 风格的 ? 占位符,同样,我们也可以使⽤ Oracle 风格的占位符:
query.prepare("INSERT INTO student (name, age) VALUES (:name, :age)");
此时,我们就需要使⽤
1 query.bindValue(":name", names);
2 query.bindValue(":age", ages);
进⾏绑定。Oracle 风格的绑定最⼤的好处是,绑定的名字和值很清晰,与顺序⽆关。但是这⾥需要注意,bindValue()函数只能绑定⼀个位置。⽐如
1 query.prepare("INSERT INTO test (name1, name2) VALUES (:name, :name)");
2// ...
3 query.bindValue(":name", name);
只能绑定第⼀个 :name 占位符,不能绑定到第⼆个。
接下来我们依旧使⽤同⼀个查询对象执⾏⼀个 SELECT 语句。如果存在查询结果,QSqlQuery::next()会返回 true,直到到达结果最末,返回 false,说明遍历结束。我们利⽤这⼀点,使⽤ while 循环即可遍历查询结果。使⽤QSqlQuery::value()函数即可按照 SELECT 语句的字段顺序获取到对应的数据库存储的数据。
对于数据库事务的操作,我们可以使⽤ QSqlDatabase::transaction() 开启事务,QSqlDatabase::commit() 或者QSqlDatabase::rollback() 结束事务。使⽤QSqlDatabase::database()函数则可以根据名字获取所需要的数据库连接。
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。
发表评论