基于Qt封裝數據庫基本增刪改查操作,支持多線程,并實現SQLite數據庫單例訪問

抽出來的,直接用就行

  • 頭文件
  • CPP文件
  • 使用示例

頭文件

#ifndef DATABASECOMMON_H
#define DATABASECOMMON_H/** 單例封裝SQLite通用操作,支持多線程調用;可擴展兼容其他數據庫,照著SysRunDatabase寫,并且重載openDatabase即可* 使用步驟:1、主線程addDatabase,調用setDatabase*         2、setDatabaseName*         3、按需setUserName、setPassword*         4、openDatabase*         5、查詢使用execSelect,建表使用execNormalSQL,插入、修改、刪除等操作使用execTransaction*         6、closeDatabase(程序生命周期內不需要顯式調用,這里僅預留)
*/#include <QObject>
#include <QSqlDatabase>
#include <QSqlQueryModel>
#include <QMutex>
/** 類名:DatabaseCommon* 功能:數據庫基礎操作抽象類,由子類實現openDatabase操作* 使用方法:子類化DatabaseCommon,并實現openDatabase函數
*/
class DatabaseCommon: public QObject
{Q_OBJECT
public:void setDatabase(QSqlDatabase db);void setDatabaseName(QString dbName);void setUserName(QString userName);void setPassword(QString password);virtual bool openDatabase() = 0;//不同數據庫的打開方式不同void closeDatabase();bool execNormalSQL(const QString &sqlString);bool execSelect(const QString &sqlString, QSqlQueryModel &model);bool execTransaction(const QString &sqlString);bool execTransaction(const QStringList &sqlStrings);
private:DatabaseCommon(const DatabaseCommon&) = delete;DatabaseCommon& operator=(const DatabaseCommon&) = delete;
protected:explicit DatabaseCommon(QObject *parent = nullptr){}virtual ~DatabaseCommon();QSqlDatabase mDatabase;QString mDatabaseName = "";QString mUserName = "";QString mPassword = "";QMutex mMutex;//保證多線程訪問安全
};
/** 類名:SysRunDatabase* 功能:應用于SysRun.db數據庫,使用全局單例進行訪問
*/
class SysRunDatabase : public DatabaseCommon
{Q_OBJECT
public:static SysRunDatabase& Instance(){static SysRunDatabase mInstance;return mInstance;}bool openDatabase() override;
protected:explicit SysRunDatabase(QObject *parent = nullptr){}~SysRunDatabase() override {}
private:SysRunDatabase(const SysRunDatabase&) = delete;SysRunDatabase& operator=(const SysRunDatabase&) = delete;
};#endif // DATABASECOMMON_H

CPP文件

#include "databasecommon.h"
#include <QDebug>
#include <QSqlQuery>
#include <QFile>
#include <QException>
#include <QSqlError>
#include <QMutexLocker>DatabaseCommon::~DatabaseCommon()
{closeDatabase();
}
/** 函數名稱:setDatabase* 功能說明:設置數據庫對象* 輸入參數:數據庫對象* 輸出參數:無* 其他說明:無*/
void DatabaseCommon::setDatabase(QSqlDatabase db)
{QMutexLocker locker(&mMutex);this->mDatabase = db;
}
/** 函數名稱:setDatabaseName* 功能說明:設置數據庫名稱(對于SQLite,就是db文件路徑)* 輸入參數:數據庫名稱* 輸出參數:無* 其他說明:無*/
void DatabaseCommon::setDatabaseName(QString dbName)
{QMutexLocker locker(&mMutex);this->mDatabaseName = dbName;
}
/** 函數名稱:setUserName* 功能說明:設置數據庫用戶名* 輸入參數:用戶名* 輸出參數:無* 其他說明:無*/
void DatabaseCommon::setUserName(QString userName)
{QMutexLocker locker(&mMutex);this->mUserName = userName;
}
/** 函數名稱:setPassword* 功能說明:設置數據庫用戶密碼* 輸入參數:用戶密碼* 輸出參數:無* 其他說明:無*/
void DatabaseCommon::setPassword(QString password)
{QMutexLocker locker(&mMutex);this->mPassword = password;
}
/** 函數名稱:openDatabase* 功能說明:打開數據庫連接* 輸入參數:無* 輸出參數:無* 其他說明:無*/
bool SysRunDatabase::openDatabase()
{QMutexLocker locker(&mMutex);if(!QFile::exists(mDatabaseName)){qCritical() << "not exist " << mDatabaseName;return false;}mDatabase.setDatabaseName(mDatabaseName);mDatabase.setUserName(mUserName);mDatabase.setPassword(mPassword);if(!mDatabase.open()){qCritical() << "open failed."<< mDatabase.lastError().text();return false;}return true;
}
/** 函數名稱:closeDatabase* 功能說明:關閉數據庫連接* 輸入參數:無* 輸出參數:無* 其他說明:無*/
void DatabaseCommon::closeDatabase()
{mDatabase.close();
}
/** 函數名稱:execNormalSQL* 功能說明:執行通用SQL語句(SELECT除外)* 輸入參數:SQL語句* 輸出參數:無* 其他說明:無*/
bool DatabaseCommon::execNormalSQL(const QString &sqlString)
{QMutexLocker locker(&mMutex);if(!mDatabase.isOpen()){qCritical() << "Database is closed.";return false;}try{QSqlQuery query(mDatabase);if (!query.exec(sqlString)){throw query.lastError();}return true;}catch (const QSqlError &error){qCritical() << "Error:" << error.text();return false;}
}
/** 函數名稱:execSelect* 功能說明:執行DQL語句(即SELECT)* 輸入參數:SELECE語句* 輸出參數:查詢模型,獲取結果* 其他說明:無*/
bool DatabaseCommon::execSelect(const QString &sqlString, QSqlQueryModel &model)
{QMutexLocker locker(&mMutex);if(!mDatabase.isOpen()){qCritical() << "Database is closed.";return false;}if(!sqlString.startsWith("SELECT", Qt::CaseInsensitive)){qCritical() << "Unsupported operation: " << sqlString;return false;}try{model.setQuery(sqlString, mDatabase);return true;}catch (const QException &ex){qCritical() << "Exception on model.setQuery: " << ex.what();qCritical() << model.lastError().text();return false;}
}
/** 函數名稱:execTransaction* 功能說明:基于事務執行SQL語句(SELETE除外)* 輸入參數:單個SQL語句* 輸出參數:無* 其他說明:無*/
bool DatabaseCommon::execTransaction(const QString &sqlString)
{QMutexLocker locker(&mMutex);if(!mDatabase.isOpen()){qCritical() << "Database is closed.";return false;}if(sqlString.startsWith("SELECT", Qt::CaseInsensitive)){qCritical() << "Unsupported operation: " << sqlString;return false;}if (!mDatabase.transaction()){qCritical() << "Failed to begin transaction:" << mDatabase.lastError().text();return false;}QSqlQuery query(mDatabase);try{if (!query.exec(sqlString)){throw query.lastError();}// 可選:檢查受影響行數if (sqlString.trimmed().startsWith("UPDATE", Qt::CaseInsensitive) && query.numRowsAffected() == 0){qWarning() << "Warning: No rows affected by:" << sqlString;}if (!mDatabase.commit()){throw mDatabase.lastError();}return true;}catch (const QSqlError &error){if (!mDatabase.rollback()){qCritical() << "Failed to rollback transaction:" << mDatabase.lastError().text();}qCritical() << "Transaction failed at command:" << query.lastQuery();qCritical() << "Error:" << error.text();return false;}
}
/** 函數名稱:execTransaction* 功能說明:基于事務執行SQL語句(SELETE除外)* 輸入參數:多個SQL語句* 輸出參數:無* 其他說明:無*/
bool DatabaseCommon::execTransaction(const QStringList &sqlStrings)
{QMutexLocker locker(&mMutex);if(!mDatabase.isOpen()){qCritical() << "Database is closed.";return false;}if (!mDatabase.transaction()){qCritical() << "Failed to begin transaction:" << mDatabase.lastError().text();return false;}QSqlQuery query(mDatabase);try{foreach (const QString &sql, sqlStrings){if (!query.exec(sql)){throw query.lastError();}// 可選:檢查受影響行數if (sql.trimmed().startsWith("UPDATE", Qt::CaseInsensitive) && query.numRowsAffected() == 0){qWarning() << "Warning: No rows affected by:" << sql;}}if (!mDatabase.commit()){throw mDatabase.lastError();}return true;}catch (const QSqlError &error){if (!mDatabase.rollback()){qCritical() << "Failed to rollback transaction:" << mDatabase.lastError().text();}qCritical() << "Transaction failed at command:" << query.lastQuery();qCritical() << "Error:" << error.text();return false;}
}

使用示例

QSqlDatabase::addDatabase("QSQLITE", "SysRun");//SysRun是數據庫連接名
SysRunDatabase::Instance().setDatabase(QSqlDatabase::database("SysRun", false));
SysRunDatabase::Instance().setDatabaseName("123.db");//123.db是數據庫文件名
/*打開數據庫連接*/
if(!SysRunDatabase::Instance().openDatabase())
{return false;
}
//在此之后就可以增刪改查

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/bicheng/83193.shtml
繁體地址,請注明出處:http://hk.pswp.cn/bicheng/83193.shtml
英文地址,請注明出處:http://en.pswp.cn/bicheng/83193.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

AI筆記 - 網絡模型 - mobileNet

網絡模型 mobileNet mobileNet V1網絡結構深度可分離卷積空間可分![在這里插入圖片描述](https://i-blog.csdnimg.cn/direct/aff06377feac40b787cfc882be7c6e5d.png) 參考 mobileNet V1 網絡結構 MobileNetV1可以理解為VGG中的標準卷積層換成深度可分離卷積 可分離卷積主要有…

第十五篇:MySQL 高級實戰項目:構建高可用、可觀測、性能優化一體化數據庫平臺

本篇聚焦于如何基于 MySQL 構建一個真正面向生產環境的數據庫平臺&#xff0c;集成高可用、可觀測與性能調優三大核心能力&#xff0c;助力穩定、可擴展的系統運行。 一、項目背景與目標 在實際生產環境中&#xff0c;數據庫系統需要應對以下挑戰&#xff1a; 業務高速增長帶來…

華為OD機試真題——文件目錄大小(2025 A卷:100分)Java/python/JavaScript/C++/C語言/GO六種語言最佳實現

2025 A卷 100分 題型 本文涵蓋詳細的問題分析、解題思路、代碼實現、代碼詳解、測試用例以及綜合分析; 并提供Java、python、JavaScript、C++、C語言、GO六種語言的最佳實現方式! 2025華為OD真題目錄+全流程解析/備考攻略/經驗分享 華為OD機試真題《文件目錄大小》: 目錄 題…

qwen 2.5 并行計算機制:依靠 PyTorch 和 Transformers 庫的分布式能力

qwen 2.5 并行計算機制:依靠 PyTorch 和 Transformers 庫的分布式能力 完整可運行代碼: import torch import torch.nn.functional as F from transformers

TIDB創建索引失敗 mkdir /tmp/tidb/tmp_ddl-4000/1370: no such file or directory.

TIDB創建索引失敗&#xff1a;解決“mkdir /tmp/tidb/tmp_ddl-4000/1370: no such file or directory”問題 在使用 TIDB 數據庫時&#xff0c;我們有時會遇到創建索引失敗的問題。常見的錯誤信息為&#xff1a; mkdir /tmp/tidb/tmp_ddl-4000/1370: no such file or directo…

華為OD機試真題—— 最少數量線段覆蓋/多線段數據壓縮(2025A卷:100分)Java/python/JavaScript/C++/C語言/GO六種最佳實現

2025 A卷 100分 題型 本文涵蓋詳細的問題分析、解題思路、代碼實現、代碼詳解、測試用例以及綜合分析; 并提供Java、python、JavaScript、C++、C語言、GO六種語言的最佳實現方式! 2025華為OD真題目錄+全流程解析/備考攻略/經驗分享 華為OD機試真題《最少數量線段覆蓋/多線段數…

EasyRTC嵌入式音視頻實時通話SDK助力AI與IoT智能硬件打造音視頻交互多場景應用

一、引言? 在數字化浪潮下&#xff0c;AI與IoT深度融合重塑智能硬件產業。實時音視頻通信是智能硬件交互的核心&#xff0c;其性能關乎用戶體驗與場景拓展。EasyRTC嵌入式音視頻實時通話SDK基于WebRTC技術&#xff0c;以輕量、易擴展的特性&#xff0c;為AI與IoT智能硬件融合…

第十四章 MQTT訂閱

系列文章目錄 系列文章目錄 第一章 總體概述 第二章 在實體機上安裝ubuntu 第三章 Windows遠程連接ubuntu 第四章 使用Docker安裝和運行EMQX 第五章 Docker卸載EMQX 第六章 EMQX客戶端MQTTX Desktop的安裝與使用 第七章 EMQX客戶端MQTTX CLI的安裝與使用 第八章 Wireshark工具…

【第4章 圖像與視頻】4.4 離屏 canvas

文章目錄 前言為什么要使用 offscreenCanvas為什么要使用 OffscreenCanvas如何使用 OffscreenCanvas第一種使用方式第二種使用方式 計算時長超過多長時間適合用Web Worker 前言 在 Canvas 開發中&#xff0c;我們經常需要處理復雜的圖形和動畫&#xff0c;這些操作可能會影響頁…

Go語言事件總線EventBus本地事件總線系統的完整實現框架

在Go語言中&#xff0c;EventBus是一種非常有用的工具&#xff0c;它通過事件驅動的編程方式&#xff0c;幫助開發者實現組件之間的解耦&#xff0c;提高代碼的可維護性和擴展性。 背景 軟件架構的發展需求&#xff1a;隨著軟件系統的規模和復雜度不斷增大&#xff0c;傳統的緊…

Go語言接口:靈活多態的核心機制

引言 Go語言的接口系統是其??面向對象編程??的核心&#xff0c;它摒棄了傳統語言的類繼承體系&#xff0c;采用獨特的??隱式實現??和??鴨子類型??設計。這種設計使得Go接口既靈活又強大&#xff0c;成為構建松耦合系統的關鍵工具。本文將深入剖析Go接口的實現機制…

DeviceNET轉EtherCAT網關:醫院藥房自動化的智能升級神經中樞

在現代醫院藥房自動化系統中&#xff0c;高效、精準、可靠的設備通信是保障患者用藥安全與效率的核心。當面臨既有支持DeviceNET協議的傳感器、執行器&#xff08;如藥盒狀態傳感器、機械臂限位開關&#xff09;需接入先進EtherCAT高速實時網絡時&#xff0c;JH-DVN-ECT疆鴻智能…

android實現使用RecyclerView詳細

顯示頁面代碼&#xff1a;activity_category_inventory.xml代碼&#xff1a; <?xml version"1.0" encoding"utf-8"?> <LinearLayout xmlns:android"http://schemas.android.com/apk/res/android" xmlns:app"http://schemas.and…

【SpringBoot實戰】優雅關閉服務

文章目錄 一、什么是優雅關閉&#xff1f;二、優雅關閉的核心步驟三、SpringBoot優雅關閉實現四、關鍵注意事項1. 超時時間必須配置2. 信號支持局限性3. 特殊請求處理 五、底層實現原理六、總結 一、什么是優雅關閉&#xff1f; 優雅關閉&#xff08;Graceful Shutdown&#x…

C++哈希表:unordered系列容器詳解

本節目標 1.unordered系列關聯式容器 2.底層結構 3.模擬實現 4.哈希的應用 5.海量數據處理面試題 unordered系列關聯式容器 在c98中&#xff0c;STL提供了底層為紅黑樹結構的一系列關聯式容器&#xff0c;在查詢時效率可以達到logN&#xff0c;即最差的情況下需要比較紅…

java操作服務器文件(把解析過的文件遷移到歷史文件夾地下)

第一步導出依賴 <dependency><groupId>org.apache.sshd</groupId><artifactId>sshd-core</artifactId><version>2.13.0</version></dependency> 第二步寫代碼 public void moveFile( List<HmAnalysisFiles> hmAnalys…

Oracle OCP認證的技術定位怎么樣?

一、引言&#xff1a;Oracle OCP認證的技術定位? Oracle Certified Professional&#xff08;OCP&#xff09;認證是數據庫領域含金量最高的國際認證之一&#xff0c;其核心價值在于培養具備企業級數據庫全生命周期管理能力的專業人才。隨著數字化轉型加速&#xff0c;OCP認證…

TK海外搶單源碼/指定卡單

? 搶單源碼&#xff0c;有指定派單&#xff0c;打針&#xff0c;這套二改過充值跳轉客服 前端vue 后端php 兩端分離 可二開 可以指定卡第幾單&#xff0c;金額多少&#xff0c; 前后端開源 PHP7.2 MySQL5.6 前端要www.域名&#xff0c;后端要admin.域名 前端直接靜態 偽靜…

遠程線程注入

注入簡單來說就是讓別人的程序執行 你想要讓他執行的dll #include<iostream> #include<Windows.h> using namespace std;char szBuffer[] "C:\\Users\\20622\\source\\repos\\Dll1\\Debug\\test.dll"; //dll路徑void RemoteThreadInject(DWORD Pid,PCH…

【Java實戰】集合排序方法與長度獲取方法辨析(易懂版)

一、排序方法 1. 對List排序的兩種方式 方式一Collections.sort() List<Integer> numbers Arrays.asList(3,1,4,2); Collections.sort(numbers); // 直接修改原list → [1,2,3,4]方式二&#xff1a;list.sort()&#xff08;Java8推薦&#xff09; List<String>…