網站首頁 編程語言 正文
一、數據庫
Qt中的Qt SQL模塊提供了對數據庫的支持,模塊中類可分為三層:驅動層,sql接口層,用戶層。
- 驅動層:(QSqlDriver,QSqlDriverCreator,QSqlDriverCreatorBase,QSqlDriverPlugin)為具體的數據庫和SQL接口層之間提供了底層的橋梁;
- SQL接口層:(QSqlDatabase,QSqlQuery,QSqlError,QSqlRecord)提供了對數據庫的訪問,其中QSqlDatabase類用來創建連接,QSqlQuery類可以使用SQL語句來實現與數據庫交互;
- 用戶接口層:(QSqlTableModel,QSqlQueryModel,QSqlRelationalTableModel)實現了將數據庫中的數據鏈接到窗口部件上,這些類是使用模型/視圖框架實現的,它們是更高層次的抽象;
1.數據庫驅動
Qt SQL模塊使用數據庫驅動插件來和不同的數據庫接口進行通信。由于Qt SQL模塊的接口是獨立于數據庫的,所以所有數據庫特定的代碼都包含在了這些驅動中。Qt默認支持一些驅動:
驅動名稱 | 數據庫 |
---|---|
QSQLITE2 | SQLite2版本 |
QSQLITE | SQLite3版本 |
QMYSQL | MySQL |
QODBC | SQL Service |
QPSQL | PostgreSQL(>=7.3版本) |
2.查詢驅動
#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.連接數據庫
數據庫連接使用連接名來定義,而不是使用數據庫名,可以向相同的數據庫創建多個連接。QSqlDatabase也支持默認連接的概念,默認連接就是一個沒有命名的連接。在使用QSqlQuery或者QSqlQueryModel的成員函數時需要指定一個連接名作為參數,如果沒有指定,那么就會使用默認連接。
原型: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();
}
創建兩個名為“first”和“second”的連接:
QSqlDatabase firstDB = QSqlDatabase::addDatabase("QMYSQL", "first");
QSqlDatabase secondDB = QSqlDatabase::addDatabase("QMYSQL", "second");
創建完連接后,可以在任何地方使用QSqlDatabase::database()靜態函數通過連接名稱獲取指向數據庫連接的指針,如果調用該函數時沒有指明連接名稱,那么會返回默認連接,例如:
QSqlDatabase defaultDB = QSqlDatabase::database();
QSqlDatabase firstDB = QSqlDatabase::database("first");
QSqlDatabase secondDB = QSqlDatabase::database("second");
要移除一個數據庫連接,需要先使用QSqlDatabase::close()關閉數據庫,然后使用靜態函數QSqlDatabase::removeDatabase()移除該連接。
連接SQL Server數據庫
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數據庫
QSqlDatabase db1 = QSqlDatabase::addDatabase("QSQLITE");
db1.setHostName("IP");
db1.setDatabaseName("dbname");
db1.setUserName("user");
db1.setPassword("password");
4.執行sql語句
QSqlQuery類提供了一個接口,用于執行SQL語句和瀏覽查詢的結果集。要執行一個SQL語句,只需要簡單的創建一個QSqlQuery對象,然后調用QSqlQuery::exec()函數即可。
QSqlQuery query;
query.exec("select * from student");
QSqlQuery提供了對結果集的訪問,可以一次訪問一條記錄。當執行完exec()函數后,QSqlQuery的內部指針會位于第一條記錄前面的位置。必須調用一次QSqlQuery::next()函數來使其前進到第一條記錄,然后可以重復使用next()函數來訪問其他的記錄,直到該函數的返回值為false,
while(query.next())
{
qDebug() << query.value(0).toInt() << query.value(1).toString();
}
在QSqlQuery類中提供了多個函數來實現在結果集中進行定位,比如next()定位到下一條記錄,previous()定位到前一條記錄,first()定位的第一條記錄,last()定位到最后一條記錄,seek(n)定位到第n條記錄。當前行的索引可以使用at()返回;record()函數可以返回當前指向的記錄。
5.插入數據
插入一條記錄
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();
當要插入多條記錄時,只需要調用QSqlQuery::prepare()一次,然后使用多次bindValue()或者addBindValue()函數來綁定需要的數據,最后調用一次exec()函數就可以了。其實,進行多條數據插入時,還可以使用批處理進行:
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.事務
事務可以保證一個復雜的操作的原子性,就是對于一個數據庫操作序列,這些操作要么全部做完,要么一條也不做,是不可分割的工作單位。如果底層的數據庫引擎支持事務,QSqlDriver::hasFeature(QSqlDriver::Transactions)會返回true。可以使用QSqlDatabase::transaction()來啟動一個事務,然后編寫希望在事務中執行的SQL語句,最后調用QSqlDatabase::commit()提交或者QSqlDatabase::rollback()回滾。使用事務必須在創建查詢以前就開始事務
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個更高層的類來訪問數據庫,分別是QSqlQueryModel、QSqlTableModel和QSqlRelationalTableModel。
這3個類都是從QAbstractTableModel派生來的,可以很容易地實現將數據庫中的數據在QListView和QTableView等項視圖類中進行顯示。使用這些類的另一個好處是,這樣可以使編寫的代碼很容易的適應其他的數據源。例如,如果開始使用了QSqlTableModel,而后來要改為使用XML文件來存儲數據,這樣需要做的僅是更換一個數據模型
1.QSqlQueryModel模型
QSqlQueryModel提供了一個基于SQL查詢的只讀模型。
QSqlQueryModel *model = new QSqlQueryModel(this);
model->setQuery("select * from student");
model->setHeaderData(0, Qt::Horizontal, tr("學號"));
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);
先創建了QSqlQueryModel對象,然后使用setQuery()來執行SQL語句查詢整張student表,并使用setHeaderData()來設置顯示的標頭。后面創建了視圖,并將QSqlQueryModel對象作為其要顯示的模型。這里要注意,其實QSqlQueryModel中存儲的是執行完setQuery()函數后的結果集,所以視圖中顯示的是結果集的內容。QSqlQueryModel中還提供了columnCount()返回一條記錄中字段的個數;rowCount()返回結果集中記錄的條數;record()返回第n條記錄;index()返回指定記錄的指定字段的索引;clear()可以清空模型中的結果集。
2.QSqlTableModel模型
QSqlTableModel提供了一個一次只能操作一個SQL表的讀寫模型,它是QSqlQuery的更高層次的替代品,可以瀏覽和修改獨立的SQL表,并且只需編寫很少的代碼,而且不需要了解SQL語法。
創建數據表
QSqlQuery query;
// 創建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)");
// 創建course表
query.exec("create table course (id int primary key, "
"name varchar, teacher varchar)");
query.exec("insert into course values(10, '數學', '王老師')");
query.exec("insert into course values(11, '英語', '張老師')");
query.exec("insert into course values(12, '計算機', '李老師')");
顯示表:
QSqlTableModel *model = new QSqlTableModel(this);
model->setTable("student");
model->select();
// 設置編輯策略
model->setEditStrategy(QSqlTableModel::OnManualSubmit);
ui->tableView->setModel(model);
ui->tableView->verticalHeader()->setHidden(true);
ui->tableView->horizontalHeader()->setSectionResizeMode(QHeaderView::Stretch);
創建一個QSqlTableModel后,只需使用setTable()來為其指定數據庫表,然后使用select()函數進行查詢,調用這兩個函數就等價于執行了“select * from student”語句。這里還可以使用setFilter()來指定查詢時的條件。在使用該模型以前,一般還要設置其編輯策略,它由QSqlTableModel::EditStrategy枚舉類型定義。
常量 | 描述 |
---|---|
QSqlTableModel::OnFieldChange | 所有對模型的改變都會立即應用到數據庫 |
QSqlTableModel::OnRowChange | 對一條記錄的改變會在用戶選擇另一條記錄時被應用 |
QSqlTableModel::OnManualSubmit | 所有的改變都會在模型中進行緩存,直到調用submitAll()或者reverAll()函數 |
修改
// 開始事務操作
model->database().transaction();
if (model->submitAll())
{
if(model->database().commit()) // 提交
QMessageBox::information(this, tr("tableModel"),tr("數據修改成功!"));
} else
{
model->database().rollback(); // 回滾
QMessageBox::warning(this, tr("tableModel"),tr("數據庫錯誤: %1").arg(model->lastError().text()),QMessageBox::Ok);
}
撤銷修
model->revertAll();
查詢
//全部數據
model->setTable("student");
model->select();
//進行篩選
QString name = "xxx" ;
// 根據姓名進行篩選,一定要使用單引號
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("刪除當前行!"),
tr("你確定刪除當前行嗎?"), QMessageBox::Yes, QMessageBox::No);
if(ok == QMessageBox::No)
{ // 如果不刪除,則撤銷
model->revertAll();
} else { // 否則提交,在數據庫中刪除該行
model->submitAll();
}
3.QSqlRelationalTableModel模型
QSqlRelationalTableModel繼承自QSqlTableModel,并且對其進行了擴展,提供了對外鍵的支持。一個外鍵就是一個表中的一個字段和其他表中的主鍵字段之間的一對一的映射。例如,student表中的course字段對應的是course表中的id字段,那么就稱字段course是一個外鍵。因為這里的course字段的值是一些數字,這樣的顯示很不友好,使用關系表格模型,就可以將它顯示為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顯示和編輯數據。這個委托為一個外鍵提供了一個QComboBox部件來顯示所有可選的數據,這樣就顯得更加清晰了。
view->setItemDelegate(new QSqlRelationalDelegate(view));
總結?
原文鏈接:https://blog.csdn.net/qq_45893999/article/details/122246874
相關推薦
- 2022-11-26 React?DnD如何處理拖拽詳解_React
- 2022-03-15 PEM_read_bio_X509_AUX() failed (SSL: error:0906D06
- 2022-02-02 appname is automatically signed for development,
- 2022-04-11 Python如何在終端彩色打印輸出_python
- 2022-12-25 Flutter開發通用頁面Loading組件示例詳解_Android
- 2022-06-10 python?PIL?Image?圖像處理基本操作實例_python
- 2023-02-09 C++命名空間?缺省參數?const總結?引用總結?內聯函數?auto關鍵字詳解_C 語言
- 2022-03-24 postman接口做關聯測試的方法步驟_相關技巧
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支