Qt入門學習之數據庫操作指南

一、數據庫

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));

總結 

到此這篇關於Qt入門學習之數據庫操作指南的文章就介紹到這瞭,更多相關Qt數據庫操作內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: