這篇文章主要介紹“Qt怎么實現(xiàn)記錄清理”,在日常操作中,相信很多人在Qt怎么實現(xiàn)記錄清理問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”Qt怎么實現(xiàn)記錄清理”的疑惑有所幫助!接下來,請跟著小編一起來學(xué)習(xí)吧!
為伊州等地區(qū)用戶提供了全套網(wǎng)頁設(shè)計制作服務(wù),及伊州網(wǎng)站建設(shè)行業(yè)解決方案。主營業(yè)務(wù)為成都網(wǎng)站制作、做網(wǎng)站、伊州網(wǎng)站設(shè)計,以傳統(tǒng)方式定制建設(shè)網(wǎng)站,并提供域名空間備案等一條龍服務(wù),秉承以專業(yè)、用心的態(tài)度為用戶提供真誠的服務(wù)。我們深信只要達(dá)到每一位用戶的要求,就會得到認(rèn)可,從而選擇與我們長期合作。這樣,我們也可以走得更遠(yuǎn)!
記錄清理功能,在數(shù)據(jù)量很小的情況下,用不上,如果數(shù)據(jù)量大了的話,長年累月存儲的,那就顯得極其重要了,好比視頻監(jiān)控中的NVR存儲的視頻一樣,一般來說存儲個60天,那超過60天怎辦呢,擦除早期的數(shù)據(jù)用來存儲最近的數(shù)據(jù)即可。在這個氣體安全管理系統(tǒng)中,數(shù)據(jù)量長年累月也是很大的,一般來說一個節(jié)點默認(rèn)每分鐘存儲一個數(shù)據(jù),如果100個節(jié)點(這個應(yīng)該是保守的數(shù)量,據(jù)說應(yīng)用的好多個現(xiàn)場有500個左右的節(jié)點,至于如何突破的modbus255個節(jié)點的限制,后面的文章會單獨講解),一天下來就是1006024=144000,100天都1440萬條數(shù)據(jù)了,千萬級別,這個如果不做早期數(shù)據(jù)的清理的話,慢慢下去各種性能影響會越來越突出,要么加大存儲間隔,要么設(shè)定只保存最近的多少條記錄,這幾個參數(shù)都是可以調(diào)節(jié)的,存儲間隔在探測器設(shè)置中進(jìn)行設(shè)置,每個探測器節(jié)點都可以設(shè)置不同的存儲間隔;保留最大的記錄數(shù)在系統(tǒng)設(shè)置中設(shè)置,默認(rèn)100W條,超過的會定時清理早期數(shù)據(jù)。
為此特定封裝了一個類DbCleanThread專門用來清理數(shù)據(jù)和文件夾,為什么還有個清理文件夾的功能呢,這也是現(xiàn)實中需求來的,比如有時候存儲一些圖片和視頻文件等,如果存儲器容量不夠大,尤其是嵌入式linux,板子上一般存儲器不會很大,目前8G居多,隨著時間的推移,文件數(shù)量會越來越多,本身存儲的剩余容量會越來越小,這也需要定期清理早期的文件,保證在現(xiàn)有容量允許的情況下,循環(huán)存儲文件,編寫這個類盡量做成了通用的類,考慮了很多個參數(shù),比如可以設(shè)定條件字段、排序字段、清理的間隔、文件夾路徑、最大大小等。
采集數(shù)據(jù)端口,支持串口端口+網(wǎng)絡(luò)端口,串口支持自由設(shè)置串口號+波特率,網(wǎng)絡(luò)支持自由設(shè)置IP地址+通訊端口,每個端口支持采集周期,默認(rèn)1秒鐘一個地址,支持設(shè)置通訊超時次數(shù),默認(rèn)3次,支持最大重連時間,用于重新讀取離線的設(shè)備。
控制器信息,能夠添加控制器名稱,選擇控制器地址+控制器型號,設(shè)置該控制器下面的探測器數(shù)量。
探測器信息,能夠添加位號,可自由選擇探測器型號,氣體種類,氣體符號,高報值,低報值,緩沖值,清零值,是否啟用,報警聲音,背景地圖,存儲周期,數(shù)值換算小數(shù)點位數(shù),報警延時時間,報警的類型(HH,LL,HL)等。
控制器型號+探測器型號+氣體種類+氣體符號,均可自由配置。
地圖支持導(dǎo)入和刪除,所有的探測器對應(yīng)地圖位置可自由拖動保存。
端口信息+控制器信息+探測器信息,支持導(dǎo)入導(dǎo)出+導(dǎo)出到excel+打印。
運行記錄+報警記錄+用戶記錄,支持多條件組合查詢,比如時間段+控制器+探測器等,所有記錄支持導(dǎo)出到excel+打印。
導(dǎo)出到excel的記錄支持所有excel+wps等表格文件版本,不依賴excel等軟件。
可刪除指定時間范圍內(nèi)的數(shù)據(jù),支持自動清理早期數(shù)據(jù),設(shè)置最大保存記錄數(shù)。
支持報警短信轉(zhuǎn)發(fā),支持多個接收手機(jī)號碼,可設(shè)定發(fā)送間隔,比如即時發(fā)送或者6個小時發(fā)送一次所有的報警信息,短信內(nèi)容過長,自動拆分多條短信。
支持報警郵件轉(zhuǎn)發(fā),支持多個接收郵箱,可設(shè)定發(fā)送間隔,比如即時發(fā)送或者6個小時發(fā)送一次所有的報警信息,支持附件發(fā)送。
高報顏色+低報顏色+正常顏色+0值顏色+曲線背景+曲線顏色等,都可以自由選擇。
軟件的中文標(biāo)題+英文標(biāo)題+logo路徑+版權(quán)所有都可以自由設(shè)置。
提供開關(guān)設(shè)置開機(jī)運行+報警聲音+自動登錄+記住密碼等。
報警聲音可設(shè)置播放次數(shù),界面提供17種皮膚文件選擇。
支持云端數(shù)據(jù)同步,可設(shè)置云端數(shù)據(jù)庫的信息,比如數(shù)據(jù)庫名稱,用戶名+密碼等。
支持網(wǎng)絡(luò)轉(zhuǎn)發(fā)和網(wǎng)絡(luò)接收,網(wǎng)絡(luò)接收開啟后,軟件從udp接收數(shù)據(jù)進(jìn)行解析。網(wǎng)絡(luò)轉(zhuǎn)發(fā)支持多個目標(biāo)IP,這樣就實現(xiàn)了本地采集的軟件,自由將數(shù)據(jù)轉(zhuǎn)到客戶端,隨時查看探測器數(shù)據(jù)。
自動記住用戶最后停留的界面+其他信息,重啟后自動應(yīng)用。
報警自動切換到對應(yīng)的地圖,探測器按鈕閃爍。
雙擊探測器圖標(biāo),可以進(jìn)行回控。
支持用戶權(quán)限管理,管理員+操作員兩大類,用戶登錄+用戶退出,可以記住密碼和自動登錄,超過三次報錯提示并關(guān)閉程序。
支持四種監(jiān)控模式,設(shè)備面板監(jiān)控+地圖監(jiān)控+表格數(shù)據(jù)監(jiān)控+曲線數(shù)據(jù)監(jiān)控,可自由切換,四種同步應(yīng)用。
支持報警繼電器聯(lián)動,一個位號可以跨串口聯(lián)動多個模塊和繼電器號,支持多對多。
本地數(shù)據(jù)存儲支持sqlite+MySQL,支持遠(yuǎn)程數(shù)據(jù)同步到云端數(shù)據(jù)庫。自動重連。
本地設(shè)備采集到的數(shù)據(jù)實時上傳到云端,以便手機(jī)APP或者web等其他方式提取。
支持兩種數(shù)據(jù)源,一種是串口和網(wǎng)絡(luò)通過協(xié)議采集設(shè)備數(shù)據(jù),一種是數(shù)據(jù)庫采集。數(shù)據(jù)庫采集模式可以作為通用的系統(tǒng)使用。
自帶設(shè)備模擬工具,支持16個設(shè)備數(shù)據(jù)模擬,同時還帶數(shù)據(jù)庫數(shù)據(jù)模擬,以便在沒有設(shè)備的時候測試數(shù)據(jù)。
默認(rèn)通信協(xié)議采用modbus協(xié)議,后期增加mqtt等物聯(lián)網(wǎng)協(xié)議的支持,做成通用系統(tǒng)。
支持所有windows操作系統(tǒng)+linux操作系統(tǒng)和其他操作系統(tǒng)。

#pragma execution_character_set("utf-8")
#include "dbcleanthread.h"
QScopedPointer<DbCleanThread> DbCleanThread::self;
DbCleanThread *DbCleanThread::Instance()
{
if (self.isNull()) {
static QMutex mutex;
QMutexLocker locker(&mutex);
if (self.isNull()) {
self.reset(new DbCleanThread);
}
}
return self.data();
}
DbCleanThread::DbCleanThread(QObject *parent) : QThread(parent)
{
stopped = false;
isBusy = false;
lastTime = QDateTime::currentDateTime();
tableName = "DeviceLog";
countName = "*";
whereColumnName = "SaveTime";
orderSql = "SaveTime asc";
maxCount = 100000;
interval = 30 * 60;
dirPath = "";
dirFileFilter = QStringList("*.jpg");
dirMaxSize = 1024;
dbType = DbType_MySql;
connName = "qt_sql_default_connection";
hostName = "127.0.0.1";
port = 3306;
dbName = "db_mysql";
userName = "root";
userPwd = "root";
}
DbCleanThread::~DbCleanThread()
{
this->stop();
this->wait();
}
void DbCleanThread::run()
{
//在這里打開數(shù)據(jù)庫,Qt5.10以后增加了安全性,不能用主線程創(chuàng)建的數(shù)據(jù)庫連接
QSqlDatabase dbConn;
if (dbType == DbType_Sqlite) {
dbConn = QSqlDatabase::addDatabase("QSQLITE", connName);
dbConn.setDatabaseName(dbName);
} else if (dbType == DbType_MySql) {
dbConn = QSqlDatabase::addDatabase("QMYSQL", connName);
dbConn.setConnectOptions("MYSQL_OPT_RECONNECT=1;MYSQL_OPT_CONNECT_TIMEOUT=1;MYSQL_OPT_READ_TIMEOUT=1;MYSQL_OPT_WRITE_TIMEOUT=1");
dbConn.setHostName(hostName);
dbConn.setPort(port);
dbConn.setDatabaseName(dbName);
dbConn.setUserName(userName);
dbConn.setPassword(userPwd);
}
//數(shù)據(jù)庫打開失敗則不用處理
if (!dbConn.open()) {
qDebug() << this->objectName() << dbConn.lastError();
return;
}
while (!stopped) {
//定時執(zhí)行一次校驗時間是否到了該清理的時候
QDateTime now = QDateTime::currentDateTime();
if (lastTime.secsTo(now) >= interval) {
if (isBusy) {
qDebug() << this->objectName() << "DbCleanThread isBusy";
lastTime = now;
continue;
}
isBusy = true;
clean();
isBusy = false;
lastTime = now;
msleep(1);
}
msleep(100);
}
stopped = false;
}
void DbCleanThread::setTableName(const QString &tableName)
{
this->tableName = tableName;
}
void DbCleanThread::setWhereColumnName(const QString &whereColumnName)
{
this->whereColumnName = whereColumnName;
}
void DbCleanThread::setOrderSql(const QString &orderSql)
{
this->orderSql = orderSql;
}
void DbCleanThread::setMaxCount(int maxCount)
{
this->maxCount = maxCount;
}
void DbCleanThread::setInterval(int interval)
{
this->interval = interval;
}
void DbCleanThread::setPar(const QString &tableName, const QString &countName,
const QString &whereColumnName, const QString &orderSql,
int maxCount, int interval)
{
this->tableName = tableName;
this->countName = countName;
this->whereColumnName = whereColumnName;
this->orderSql = orderSql;
this->maxCount = maxCount;
this->interval = interval;
}
void DbCleanThread::setDirPath(const QString &dirPath)
{
this->dirPath = dirPath;
}
void DbCleanThread::setDirFileFilter(const QStringList &dirFileFilter)
{
this->dirFileFilter = dirFileFilter;
}
void DbCleanThread::setDirMaxSize(int dirMaxSize)
{
this->dirMaxSize = dirMaxSize;
}
void DbCleanThread::setConnInfo(DbCleanThread::DbType dbType, const QString &connName, const QString &hostName, int port,
const QString &dbName, const QString &userName, const QString &userPwd)
{
this->dbType = dbType;
this->connName = connName;
this->hostName = hostName;
this->port = port;
this->dbName = dbName;
this->userName = userName;
this->userPwd = userPwd;
}
void DbCleanThread::deleteDirectory(const QString &path)
{
QDir dir(path);
if (!dir.exists()) {
return;
}
dir.setFilter(QDir::AllEntries | QDir::NoDotAndDotDot);
QFileInfoList fileList = dir.entryInfoList();
foreach (QFileInfo fi, fileList) {
if (fi.isFile()) {
fi.dir().remove(fi.fileName());
} else {
deleteDirectory(fi.absoluteFilePath());
dir.rmdir(fi.absoluteFilePath());
}
}
}
void DbCleanThread::stop()
{
stopped = true;
}
void DbCleanThread::clean()
{
//首先查找總記錄數(shù),如果總記錄數(shù)超過限制,則將超出的部分按照字段排序進(jìn)行刪除
QString sql = QString("select count(%1) from %2").arg(countName).arg(tableName);
QSqlQuery query(QSqlDatabase::database(connName));
query.exec(sql);
query.next();
int count = query.value(0).toInt();
int cleanCount = (count - maxCount);
if (cleanCount >= 100) {
QDateTime dtStart = QDateTime::currentDateTime();
//每次最大清理1000條數(shù)據(jù)
cleanCount = cleanCount > 1000 ? 1000 : cleanCount;
//將要刪除的數(shù)據(jù)指定字段集合查詢出來
query.clear();
sql = QString("select %2 from %1 order by %3 limit %4").arg(tableName).arg(whereColumnName).arg(orderSql).arg(cleanCount);
query.exec(sql);
QStringList list;
while(query.next()) {
list << query.value(0).toString();
}
//刪除數(shù)據(jù)
query.clear();
sql = QString("delete from %1 where %2 in(%3)").arg(tableName).arg(whereColumnName).arg(list.join(","));
bool ok = query.exec(sql);
QDateTime dtEnd = QDateTime::currentDateTime();
double msec = dtStart.msecsTo(dtEnd);
emit cleanFinsh(ok, cleanCount, msec);
}
//自動清理文件夾
if (!dirPath.isEmpty()) {
//找出該文件夾下的所有文件夾
QDir dir(dirPath);
if (!dir.exists()) {
return;
}
//按照目錄查找,過濾文件夾,按照文件名稱排序
dir.setFilter(QDir::Dirs | QDir::NoSymLinks | QDir::NoDotAndDotDot);
dir.setSorting(QDir::Name);
QStringList list = dir.entryList();
//遍歷所有目錄,對所有文件大小相加得到總大小,文件就在文件夾下,不會再有子目錄
qint64 size = 0;
foreach (QString path, list) {
QDir d(dirPath + "/" + path);
QFileInfoList infos = d.entryInfoList(dirFileFilter);
foreach (QFileInfo info, infos) {
size += info.size();
}
//轉(zhuǎn)化成MB,超過預(yù)定大小自動刪除第一個文件夾,跳出循環(huán)無需繼續(xù)判斷
int sizeMB = size / (1024 * 1024);
if (sizeMB >= dirMaxSize) {
//刪除該目錄下的所有文件
QString firstDir = dirPath + "/" + list.at(0);
QDir dir(firstDir);
dir.setFilter(QDir::AllEntries | QDir::NoDotAndDotDot);
QStringList list = dir.entryList();
foreach (QString file, list) {
dir.remove(file);
qDebug() << "remove file" << firstDir << file;
}
//刪除文件夾本身
dir.rmdir(firstDir);
qDebug() << "remove dir" << firstDir;
emit cleanDir(firstDir);
break;
}
}
}
}到此,關(guān)于“Qt怎么實現(xiàn)記錄清理”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識,請繼續(xù)關(guān)注創(chuàng)新互聯(lián)網(wǎng)站,小編會繼續(xù)努力為大家?guī)砀鄬嵱玫奈恼拢?/p>
名稱欄目:Qt怎么實現(xiàn)記錄清理
網(wǎng)站鏈接:http://chinadenli.net/article34/ipsdse.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供電子商務(wù)、定制網(wǎng)站、靜態(tài)網(wǎng)站、品牌網(wǎng)站建設(shè)、外貿(mào)建站、商城網(wǎng)站
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)