網(wǎng)站首頁 編程語言 正文
一、數(shù)據(jù)庫
Qt中的Qt SQL模塊提供了對(duì)數(shù)據(jù)庫的支持,模塊中類可分為三層:驅(qū)動(dòng)層,sql接口層,用戶層。
- 驅(qū)動(dòng)層:(QSqlDriver,QSqlDriverCreator,QSqlDriverCreatorBase,QSqlDriverPlugin)為具體的數(shù)據(jù)庫和SQL接口層之間提供了底層的橋梁;
- SQL接口層:(QSqlDatabase,QSqlQuery,QSqlError,QSqlRecord)提供了對(duì)數(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ū)動(dòng)
Qt SQL模塊使用數(shù)據(jù)庫驅(qū)動(dòng)插件來和不同的數(shù)據(jù)庫接口進(jìn)行通信。由于Qt SQL模塊的接口是獨(dú)立于數(shù)據(jù)庫的,所以所有數(shù)據(jù)庫特定的代碼都包含在了這些驅(qū)動(dòng)中。Qt默認(rèn)支持一些驅(qū)動(dòng):
驅(qū)動(dòng)名稱 | 數(shù)據(jù)庫 |
---|---|
QSQLITE2 | SQLite2版本 |
QSQLITE | SQLite3版本 |
QMYSQL | MySQL |
QODBC | SQL Service |
QPSQL | PostgreSQL(>=7.3版本) |
2.查詢驅(qū)動(dòng)
#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)建多個(gè)連接。QSqlDatabase也支持默認(rèn)連接的概念,默認(rèn)連接就是一個(gè)沒有命名的連接。在使用QSqlQuery或者QSqlQueryModel的成員函數(shù)時(shí)需要指定一個(gè)連接名作為參數(shù),如果沒有指定,那么就會(huì)使用默認(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)建兩個(gè)名為“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ù)時(shí)沒有指明連接名稱,那么會(huì)返回默認(rèn)連接,例如:
QSqlDatabase defaultDB = QSqlDatabase::database();
QSqlDatabase firstDB = QSqlDatabase::database("first");
QSqlDatabase secondDB = QSqlDatabase::database("second");
要移除一個(gè)數(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類提供了一個(gè)接口,用于執(zhí)行SQL語句和瀏覽查詢的結(jié)果集。要執(zhí)行一個(gè)SQL語句,只需要簡單的創(chuàng)建一個(gè)QSqlQuery對(duì)象,然后調(diào)用QSqlQuery::exec()函數(shù)即可。
QSqlQuery query;
query.exec("select * from student");
QSqlQuery提供了對(duì)結(jié)果集的訪問,可以一次訪問一條記錄。當(dāng)執(zhí)行完exec()函數(shù)后,QSqlQuery的內(nèi)部指針會(huì)位于第一條記錄前面的位置。必須調(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類中提供了多個(gè)函數(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')");
同一時(shí)間插入多條記錄,可以使用占位符來完成。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)要插入多條記錄時(shí),只需要調(diào)用QSqlQuery::prepare()一次,然后使用多次bindValue()或者addBindValue()函數(shù)來綁定需要的數(shù)據(jù),最后調(diào)用一次exec()函數(shù)就可以了。其實(shí),進(jìn)行多條數(shù)據(jù)插入時(shí),還可以使用批處理進(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ù)可以保證一個(gè)復(fù)雜的操作的原子性,就是對(duì)于一個(gè)數(shù)據(jù)庫操作序列,這些操作要么全部做完,要么一條也不做,是不可分割的工作單位。如果底層的數(shù)據(jù)庫引擎支持事務(wù),QSqlDriver::hasFeature(QSqlDriver::Transactions)會(huì)返回true。可以使用QSqlDatabase::transaction()來啟動(dòng)一個(gè)事務(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個(gè)更高層的類來訪問數(shù)據(jù)庫,分別是QSqlQueryModel、QSqlTableModel和QSqlRelationalTableModel。
這3個(gè)類都是從QAbstractTableModel派生來的,可以很容易地實(shí)現(xiàn)將數(shù)據(jù)庫中的數(shù)據(jù)在QListView和QTableView等項(xiàng)視圖類中進(jìn)行顯示。使用這些類的另一個(gè)好處是,這樣可以使編寫的代碼很容易的適應(yīng)其他的數(shù)據(jù)源。例如,如果開始使用了QSqlTableModel,而后來要改為使用XML文件來存儲(chǔ)數(shù)據(jù),這樣需要做的僅是更換一個(gè)數(shù)據(jù)模型
1.QSqlQueryModel模型
QSqlQueryModel提供了一個(gè)基于SQL查詢的只讀模型。
QSqlQueryModel *model = new QSqlQueryModel(this);
model->setQuery("select * from student");
model->setHeaderData(0, Qt::Horizontal, tr("學(xué)號(hào)"));
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對(duì)象,然后使用setQuery()來執(zhí)行SQL語句查詢整張student表,并使用setHeaderData()來設(shè)置顯示的標(biāo)頭。后面創(chuàng)建了視圖,并將QSqlQueryModel對(duì)象作為其要顯示的模型。這里要注意,其實(shí)QSqlQueryModel中存儲(chǔ)的是執(zhí)行完setQuery()函數(shù)后的結(jié)果集,所以視圖中顯示的是結(jié)果集的內(nèi)容。QSqlQueryModel中還提供了columnCount()返回一條記錄中字段的個(gè)數(shù);rowCount()返回結(jié)果集中記錄的條數(shù);record()返回第n條記錄;index()返回指定記錄的指定字段的索引;clear()可以清空模型中的結(jié)果集。
2.QSqlTableModel模型
QSqlTableModel提供了一個(gè)一次只能操作一個(gè)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)建一個(gè)QSqlTableModel后,只需使用setTable()來為其指定數(shù)據(jù)庫表,然后使用select()函數(shù)進(jìn)行查詢,調(diào)用這兩個(gè)函數(shù)就等價(jià)于執(zhí)行了“select * from student”語句。這里還可以使用setFilter()來指定查詢時(shí)的條件。在使用該模型以前,一般還要設(shè)置其編輯策略,它由QSqlTableModel::EditStrategy枚舉類型定義。
常量 | 描述 |
---|---|
QSqlTableModel::OnFieldChange | 所有對(duì)模型的改變都會(huì)立即應(yīng)用到數(shù)據(jù)庫 |
QSqlTableModel::OnRowChange | 對(duì)一條記錄的改變會(huì)在用戶選擇另一條記錄時(shí)被應(yīng)用 |
QSqlTableModel::OnManualSubmit | 所有的改變都會(huì)在模型中進(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ù)庫錯(cuò)誤: %1").arg(model->lastError().text()),QMessageBox::Ok);
}
撤銷修
model->revertAll();
查詢
//全部數(shù)據(jù)
model->setTable("student");
model->select();
//進(jìn)行篩選
QString name = "xxx" ;
// 根據(jù)姓名進(jìn)行篩選,一定要使用單引號(hào)
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,并且對(duì)其進(jìn)行了擴(kuò)展,提供了對(duì)外鍵的支持。一個(gè)外鍵就是一個(gè)表中的一個(gè)字段和其他表中的主鍵字段之間的一對(duì)一的映射。例如,student表中的course字段對(duì)應(yīng)的是course表中的id字段,那么就稱字段course是一個(gè)外鍵。因?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中還提供了一個(gè)QSqlRelationalDelegate委托類,它可以為QSqlRelationalTableModel顯示和編輯數(shù)據(jù)。這個(gè)委托為一個(gè)外鍵提供了一個(gè)QComboBox部件來顯示所有可選的數(shù)據(jù),這樣就顯得更加清晰了。
view->setItemDelegate(new QSqlRelationalDelegate(view));
總結(jié)?
原文鏈接:https://blog.csdn.net/qq_45893999/article/details/122246874
相關(guān)推薦
- 2022-06-29 python人工智能tensorflow函數(shù)tf.assign使用方法_python
- 2022-09-03 Python實(shí)現(xiàn)計(jì)算AUC的示例代碼_python
- 2022-06-18 C++?詳細(xì)講解對(duì)象的構(gòu)造順序_C 語言
- 2022-07-20 Spring利用zouzhiy-excel實(shí)現(xiàn)自定義表尾導(dǎo)出
- 2022-01-12 npm ERR 部署meteor時(shí)npm install出錯(cuò)問題解決
- 2022-12-14 Android?MaterialAlertDialogBuilder修改按鈕屬性_Android
- 2023-07-15 css背景顏色不顯示
- 2022-08-05 Entity?Framework映射TPH、TPT、TPC與繼承類_C#教程
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運(yùn)行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲(chǔ)小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運(yùn)算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯(cuò)誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實(shí)現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡單動(dòng)態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對(duì)象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊(duì)列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支