Qt入門學(xué)習(xí)之?dāng)?shù)據(jù)庫操作指南
一、數(shù)據(jù)庫
Qt中的Qt SQL模塊提供了對數(shù)據(jù)庫的支持,模塊中類可分為三層:驅(qū)動層,sql接口層,用戶層。
- 驅(qū)動層:(QSqlDriver,QSqlDriverCreator,QSqlDriverCreatorBase,QSqlDriverPlugin)為具體的數(shù)據(jù)庫和SQL接口層之間提供了底層的橋梁;
- SQL接口層:(QSqlDatabase,QSqlQuery,QSqlError,QSqlRecord)提供了對數(shù)據(jù)庫的訪問,其中QSqlDatabase類用來創(chuàng)建連接,QSqlQuery類可以使用SQL語句來實(shí)現(xiàn)與數(shù)據(jù)庫交互;
- 用戶接口層:(QSqlTableModel,QSqlQueryModel,QSqlRelationalTableModel)實(shí)現(xiàn)了將數(shù)據(jù)庫中的數(shù)據(jù)鏈接到窗口部件上,這些類是使用模型/視圖框架實(shí)現(xiàn)的,它們是更高層次的抽象;
1.數(shù)據(jù)庫驅(qū)動
Qt SQL模塊使用數(shù)據(jù)庫驅(qū)動插件來和不同的數(shù)據(jù)庫接口進(jìn)行通信。由于Qt SQL模塊的接口是獨(dú)立于數(shù)據(jù)庫的,所以所有數(shù)據(jù)庫特定的代碼都包含在了這些驅(qū)動中。Qt默認(rèn)支持一些驅(qū)動:
驅(qū)動名稱 | 數(shù)據(jù)庫 |
---|---|
QSQLITE2 | SQLite2版本 |
QSQLITE | SQLite3版本 |
QMYSQL | MySQL |
QODBC | SQL Service |
QPSQL | PostgreSQL(>=7.3版本) |
2.查詢驅(qū)動
#include <QApplication> #include <QSqlDatabase> #include <QDebug> #include <QStringList> int main(int argc, char *argv[]) { QApplication a(argc, argv); QStringList drivers = QSqlDatabase::drivers(); foreach(QString driver, drivers) qDebug() << driver; return a.exec(); }
在.pro文件中加入sql模塊
3.連接數(shù)據(jù)庫
數(shù)據(jù)庫連接使用連接名來定義,而不是使用數(shù)據(jù)庫名,可以向相同的數(shù)據(jù)庫創(chuàng)建多個連接。QSqlDatabase也支持默認(rèn)連接的概念,默認(rèn)連接就是一個沒有命名的連接。在使用QSqlQuery或者QSqlQueryModel的成員函數(shù)時需要指定一個連接名作為參數(shù),如果沒有指定,那么就會使用默認(rèn)連接。
原型:QSqlDatabase QSqlDatabase::addDatabase(const QString &type, const
QString &connectionName = QLatin1String( defaultConnection ))
QSqlDatabase db = QSqlDatabase::addDatabase("QMYSQL"); db.setHostName("127.0.0.1"); db.setDatabaseName("book"); db.setUserName("root"); db.setPassword("123456"); if(!db.open()) { qDebug() << "fail to connect mysql" << db.lastError().text(); }
創(chuàng)建兩個名為“first”和“second”的連接:
QSqlDatabase firstDB = QSqlDatabase::addDatabase("QMYSQL", "first"); QSqlDatabase secondDB = QSqlDatabase::addDatabase("QMYSQL", "second");
創(chuàng)建完連接后,可以在任何地方使用QSqlDatabase::database()靜態(tài)函數(shù)通過連接名稱獲取指向數(shù)據(jù)庫連接的指針,如果調(diào)用該函數(shù)時沒有指明連接名稱,那么會返回默認(rèn)連接,例如:
QSqlDatabase defaultDB = QSqlDatabase::database(); QSqlDatabase firstDB = QSqlDatabase::database("first"); QSqlDatabase secondDB = QSqlDatabase::database("second");
要移除一個數(shù)據(jù)庫連接,需要先使用QSqlDatabase::close()關(guān)閉數(shù)據(jù)庫,然后使用靜態(tài)函數(shù)QSqlDatabase::removeDatabase()移除該連接。
連接SQL Server數(shù)據(jù)庫
QSqlDatabase db = QSqlDatabase::addDatabase("QODBC"); db.setDatabaseName(QString("DRIVER={SQL SERVER}SERVER=%1;DATABASE=%2;UID=%3;PWD=%4;") .arg("IP").arg("dbname").arg("user").arg("password")); bool ok =db.open();
連接SQLite數(shù)據(jù)庫
QSqlDatabase db1 = QSqlDatabase::addDatabase("QSQLITE"); db1.setHostName("IP"); db1.setDatabaseName("dbname"); db1.setUserName("user"); db1.setPassword("password");
4.執(zhí)行sql語句
QSqlQuery類提供了一個接口,用于執(zhí)行SQL語句和瀏覽查詢的結(jié)果集。要執(zhí)行一個SQL語句,只需要簡單的創(chuàng)建一個QSqlQuery對象,然后調(diào)用QSqlQuery::exec()函數(shù)即可。
QSqlQuery query; query.exec("select * from student");
QSqlQuery提供了對結(jié)果集的訪問,可以一次訪問一條記錄。當(dāng)執(zhí)行完exec()函數(shù)后,QSqlQuery的內(nèi)部指針會位于第一條記錄前面的位置。必須調(diào)用一次QSqlQuery::next()函數(shù)來使其前進(jìn)到第一條記錄,然后可以重復(fù)使用next()函數(shù)來訪問其他的記錄,直到該函數(shù)的返回值為false,
while(query.next()) { qDebug() << query.value(0).toInt() << query.value(1).toString(); }
在QSqlQuery類中提供了多個函數(shù)來實(shí)現(xiàn)在結(jié)果集中進(jìn)行定位,比如next()定位到下一條記錄,previous()定位到前一條記錄,first()定位的第一條記錄,last()定位到最后一條記錄,seek(n)定位到第n條記錄。當(dāng)前行的索引可以使用at()返回;record()函數(shù)可以返回當(dāng)前指向的記錄。
5.插入數(shù)據(jù)
插入一條記錄
query.exec("insert into student (id, name) values (1, 'LI')");
同一時間插入多條記錄,可以使用占位符來完成。Qt支持兩種占位符:名稱綁定和位置綁定。
1名稱綁定
query.prepare("insert into student (id, name) values (:id, :name)"); int idValue = 1; QString nameValue = "Li"; query.bindValue(":id", idValue); query.bindValue(":name", nameValue); query.exec();
2位置綁定
query.prepare("insert into student (id, name) values (?, ?)"); int idValue = 1; QString nameValue = "Li"; query.addBindValue(idValue); query.addBindValue(nameValue); query.exec();
當(dāng)要插入多條記錄時,只需要調(diào)用QSqlQuery::prepare()一次,然后使用多次bindValue()或者addBindValue()函數(shù)來綁定需要的數(shù)據(jù),最后調(diào)用一次exec()函數(shù)就可以了。其實(shí),進(jìn)行多條數(shù)據(jù)插入時,還可以使用批處理進(jìn)行:
query.prepare("insert into student (id, name) values (?, ?)"); QVariantList ids; ids << 1 << 2 << 3; query.addBindValue(ids); QVariantList names; names << "Li" << "Wang" << "Liu"; query.addBindValue(names); if(!query.execBatch()) qDebug() << query.lastError();
6.事務(wù)
事務(wù)可以保證一個復(fù)雜的操作的原子性,就是對于一個數(shù)據(jù)庫操作序列,這些操作要么全部做完,要么一條也不做,是不可分割的工作單位。如果底層的數(shù)據(jù)庫引擎支持事務(wù),QSqlDriver::hasFeature(QSqlDriver::Transactions)會返回true??梢允褂肣SqlDatabase::transaction()來啟動一個事務(wù),然后編寫希望在事務(wù)中執(zhí)行的SQL語句,最后調(diào)用QSqlDatabase::commit()提交或者QSqlDatabase::rollback()回滾。使用事務(wù)必須在創(chuàng)建查詢以前就開始事務(wù)
QSqlDatabase::database().transaction(); QSqlQuery query; query.exec("SELECT id FROM student WHERE name = 'Li'"); if (query.next()) { int id = query.value(0).toInt(); query.exec("INSERT INTO project (id, name, ownerid) " "VALUES (201, 'MProject', " + QString::number(id) + ')'); } QSqlDatabase::database().commit();
二 ,sql模型類
Qt還提供了3個更高層的類來訪問數(shù)據(jù)庫,分別是QSqlQueryModel、QSqlTableModel和QSqlRelationalTableModel。
這3個類都是從QAbstractTableModel派生來的,可以很容易地實(shí)現(xiàn)將數(shù)據(jù)庫中的數(shù)據(jù)在QListView和QTableView等項(xiàng)視圖類中進(jìn)行顯示。使用這些類的另一個好處是,這樣可以使編寫的代碼很容易的適應(yīng)其他的數(shù)據(jù)源。例如,如果開始使用了QSqlTableModel,而后來要改為使用XML文件來存儲數(shù)據(jù),這樣需要做的僅是更換一個數(shù)據(jù)模型
1.QSqlQueryModel模型
QSqlQueryModel提供了一個基于SQL查詢的只讀模型。
QSqlQueryModel *model = new QSqlQueryModel(this); model->setQuery("select * from student"); model->setHeaderData(0, Qt::Horizontal, tr("學(xué)號")); model->setHeaderData(1, Qt::Horizontal, tr("姓名")); model->setHeaderData(2, Qt::Horizontal, tr("課程")); ui->tableView->setModel(model); ui->tableView->verticalHeader()->setHidden(true); ui->tableView->horizontalHeader()->setSectionResizeMode(QHeaderView::Stretch);
先創(chuàng)建了QSqlQueryModel對象,然后使用setQuery()來執(zhí)行SQL語句查詢整張student表,并使用setHeaderData()來設(shè)置顯示的標(biāo)頭。后面創(chuàng)建了視圖,并將QSqlQueryModel對象作為其要顯示的模型。這里要注意,其實(shí)QSqlQueryModel中存儲的是執(zhí)行完setQuery()函數(shù)后的結(jié)果集,所以視圖中顯示的是結(jié)果集的內(nèi)容。QSqlQueryModel中還提供了columnCount()返回一條記錄中字段的個數(shù);rowCount()返回結(jié)果集中記錄的條數(shù);record()返回第n條記錄;index()返回指定記錄的指定字段的索引;clear()可以清空模型中的結(jié)果集。
2.QSqlTableModel模型
QSqlTableModel提供了一個一次只能操作一個SQL表的讀寫模型,它是QSqlQuery的更高層次的替代品,可以瀏覽和修改獨(dú)立的SQL表,并且只需編寫很少的代碼,而且不需要了解SQL語法。
創(chuàng)建數(shù)據(jù)表
QSqlQuery query; // 創(chuàng)建student表 query.exec("create table student (id int primary key, " "name varchar, course int)"); query.exec("insert into student values(1, '李', 10)"); query.exec("insert into student values(2, '馬', 11)"); query.exec("insert into student values(3, '孫', 12)"); // 創(chuàng)建course表 query.exec("create table course (id int primary key, " "name varchar, teacher varchar)"); query.exec("insert into course values(10, '數(shù)學(xué)', '王老師')"); query.exec("insert into course values(11, '英語', '張老師')"); query.exec("insert into course values(12, '計(jì)算機(jī)', '李老師')");
顯示表:
QSqlTableModel *model = new QSqlTableModel(this); model->setTable("student"); model->select(); // 設(shè)置編輯策略 model->setEditStrategy(QSqlTableModel::OnManualSubmit); ui->tableView->setModel(model); ui->tableView->verticalHeader()->setHidden(true); ui->tableView->horizontalHeader()->setSectionResizeMode(QHeaderView::Stretch);
創(chuàng)建一個QSqlTableModel后,只需使用setTable()來為其指定數(shù)據(jù)庫表,然后使用select()函數(shù)進(jìn)行查詢,調(diào)用這兩個函數(shù)就等價(jià)于執(zhí)行了“select * from student”語句。這里還可以使用setFilter()來指定查詢時的條件。在使用該模型以前,一般還要設(shè)置其編輯策略,它由QSqlTableModel::EditStrategy枚舉類型定義。
常量 | 描述 |
---|---|
QSqlTableModel::OnFieldChange | 所有對模型的改變都會立即應(yīng)用到數(shù)據(jù)庫 |
QSqlTableModel::OnRowChange | 對一條記錄的改變會在用戶選擇另一條記錄時被應(yīng)用 |
QSqlTableModel::OnManualSubmit | 所有的改變都會在模型中進(jìn)行緩存,直到調(diào)用submitAll()或者reverAll()函數(shù) |
修改
// 開始事務(wù)操作 model->database().transaction(); if (model->submitAll()) { if(model->database().commit()) // 提交 QMessageBox::information(this, tr("tableModel"),tr("數(shù)據(jù)修改成功!")); } else { model->database().rollback(); // 回滾 QMessageBox::warning(this, tr("tableModel"),tr("數(shù)據(jù)庫錯誤: %1").arg(model->lastError().text()),QMessageBox::Ok); }
撤銷修
model->revertAll();
查詢
//全部數(shù)據(jù) model->setTable("student"); model->select(); //進(jìn)行篩選 QString name = "xxx" ; // 根據(jù)姓名進(jìn)行篩選,一定要使用單引號 model->setFilter(QString("name = '%1'").arg(name)); model->select();
升序
//id字段,即第0列,升序排列 model->setSort(0, Qt::AscendingOrder); model->select();
刪除
// 獲取選中的行 int curRow = ui->tableView->currentIndex().row(); // 刪除該行 model->removeRow(curRow); int ok1 = QMessageBox::warning(this,tr("刪除當(dāng)前行!"), tr("你確定刪除當(dāng)前行嗎?"), QMessageBox::Yes, QMessageBox::No); if(ok == QMessageBox::No) { // 如果不刪除,則撤銷 model->revertAll(); } else { // 否則提交,在數(shù)據(jù)庫中刪除該行 model->submitAll(); }
3.QSqlRelationalTableModel模型
QSqlRelationalTableModel繼承自QSqlTableModel,并且對其進(jìn)行了擴(kuò)展,提供了對外鍵的支持。一個外鍵就是一個表中的一個字段和其他表中的主鍵字段之間的一對一的映射。例如,student表中的course字段對應(yīng)的是course表中的id字段,那么就稱字段course是一個外鍵。因?yàn)檫@里的course字段的值是一些數(shù)字,這樣的顯示很不友好,使用關(guān)系表格模型,就可以將它顯示為course表中的name字段的值。
QSqlRelationalTableModel *model = new QSqlRelationalTableModel(this); model->setTable("student"); model->setRelation(2, QSqlRelation("course", "id", "name")); model->select(); ui->tableView->setModel(model); ui->tableView->verticalHeader()->setHidden(true); ui->tableView->horizontalHeader()->setSectionResizeMode(QHeaderView::Stretch);
Qt中還提供了一個QSqlRelationalDelegate委托類,它可以為QSqlRelationalTableModel顯示和編輯數(shù)據(jù)。這個委托為一個外鍵提供了一個QComboBox部件來顯示所有可選的數(shù)據(jù),這樣就顯得更加清晰了。
view->setItemDelegate(new QSqlRelationalDelegate(view));
總結(jié)
到此這篇關(guān)于Qt入門學(xué)習(xí)之?dāng)?shù)據(jù)庫操作指南的文章就介紹到這了,更多相關(guān)Qt數(shù)據(jù)庫操作內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- QT連接MYSQL數(shù)據(jù)庫的詳細(xì)步驟
- Qt5連接并操作PostgreSQL數(shù)據(jù)庫的實(shí)現(xiàn)示例
- QT連接Oracle數(shù)據(jù)庫并實(shí)現(xiàn)登錄驗(yàn)證的操作步驟
- Qt連接MySQL數(shù)據(jù)庫的實(shí)現(xiàn)(保姆級成功版教程)
- Qt創(chuàng)建SQlite數(shù)據(jù)庫的示例代碼
- 通過Qt連接OpenGauss數(shù)據(jù)庫的詳細(xì)教程
- QT連接Mysql數(shù)據(jù)庫的實(shí)現(xiàn)步驟
- Qt操作SQLite數(shù)據(jù)庫的教程詳解
- 一篇文章詳解Qt中如何訪問數(shù)據(jù)庫
相關(guān)文章
C++實(shí)現(xiàn)循環(huán)隊(duì)列和鏈?zhǔn)疥?duì)列的示例
下面小編就為大家分享一篇C++實(shí)現(xiàn)循環(huán)隊(duì)列和鏈?zhǔn)疥?duì)列的示例,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2017-12-12C++編程之CString、string與、char數(shù)組的轉(zhuǎn)換
這篇文章主要介紹了C++編程之CString、string與、char數(shù)組的轉(zhuǎn)換的相關(guān)資料,希望通過本文能幫助到大家,讓大家學(xué)習(xí)理解這部分內(nèi)容,需要的朋友可以參考下2017-10-10C語言實(shí)現(xiàn)校運(yùn)動會項(xiàng)目管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C語言實(shí)現(xiàn)校運(yùn)動會項(xiàng)目管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-02-02二維指針動態(tài)分配內(nèi)存連續(xù)問題深入分析
當(dāng)我們定義一個二維指針時,如果需要存儲相應(yīng)的數(shù)據(jù),就需要我們動態(tài)的分配內(nèi)存,這時,有一點(diǎn)是需要注意的,分配內(nèi)存的方法不同,內(nèi)存的連續(xù)性也是不相同的2013-07-07從匯編看c++中默認(rèn)構(gòu)造函數(shù)的使用分析
c++中,如果為一個類沒有明確定義一個構(gòu)造函數(shù),那么,編譯器就會自動合成一個默認(rèn)的構(gòu)造函數(shù)。下面,通過匯編程序,來看一下其真實(shí)情況2013-05-05