QTableView在显示大量数据时,即使在子线程中对数据模型进行数据更改,依然会导致界面的卡顿,而且数据量较大时,会占用大量的内存,为解决此问题,可以在建立自定义的数据模型,替代QStandardItemModel,并在数据模型中存储需要显示的数据,但是在界面中只显示部分数据,同时监测用户操作,当需要刷新数据时,对数据进行重新加载。
核心思路即是:由于QTableView 一次能够显示的数据有限,所以只在QTableView显示部分数据,当用户操作时,更新数据。
需要用到的知识:
- 多线程;
- Qt的Model/View框架;
代码如下:
自定义的MyTableView,继承自QTableView,以实现对键盘事件的响应
头文件:
class MyTableView : public QTableView
{
Q_OBJECT
public:
MyTableView(QWidget *parent);
void wheelEvent(QWheelEvent *event);
public slots:
void on_verscrollBarChange(int value);
signals:
void verticalbarValue(double value);
void rowchanged(int row);
};
源文件:
MyTableView::MyTableView(QWidget *parent):QTableView(parent)
{
this->verticalScrollBar()->setMaximum(100);
connect(this->verticalScrollBar(),SIGNAL(valueChanged(int)),this,SLOT(on_verscrollBarChange(int)));
}
void MyTableView::wheelEvent(QWheelEvent *event)
{
int delta= event->delta();
int row;
if(delta>0)
{
row = 6;
}
else
{
row=-6;
}
emit rowchanged(row);
}
void MyTableView::on_verscrollBarChange(int value)
{
int tem = this->verticalScrollBar()->maximum();
double ratio = double(value)/tem;
emit verticalbarValue(ratio);
}
自定义的数据模型:
头文件:
#ifndef MTABLEVIEW_H
#define MTABLEVIEW_H
#include <QObject>
#include <QStandardItemModel>
#include <QStringList>
#include <QList>
struct strData{
QString str1;
QString str2;
QString str3;
QString str4;
QString str5;
QString str6;
QString str7;
QString str8;
QString str9;
QString str0;
};
class MTableModel : public QStandardItemModel
{
Q_OBJECT
public:
QList<QStringList> m_DataList;
int CurrentRow;
int TotalRows;
MTableModel();
void updateViewData();
//~MTableModel();
int GetCurrentRow();
public slots:
void on_verscrollBar(double value);
void ReceiverChanged(int row);
void LoadData(QString i_path);
};
#endif // MTABLEVIEW_H
源文件:
#include "mtablemodel.h"
#include <QFile>
#include <QIODevice>
#include <QTextStream>
#include <QDebug>
MTableModel::MTableModel()
{
TotalRows = 50;
}
void MTableModel::LoadData(QString i_path)
{
QFile file(i_path);
if (!file.open(QIODevice::ReadOnly | QIODevice::Text))
return;
int i=0;
while (!file.atEnd()) {
QString strLine=file.readLine();
QStringList strList=strLine.split(" ");
m_DataList.append(strList);
if(i<TotalRows)
{
this->setItem(i,0,new QStandardItem(strList[0]));
this->setItem(i,1,new QStandardItem(strList[1]));
this->setItem(i,2,new QStandardItem(strList[2]));
this->setItem(i,3,new QStandardItem(strList[3]));
this->setItem(i,4,new QStandardItem(strList[4]));
this->setItem(i,5,new QStandardItem(strList[5]));
this->setItem(i,6,new QStandardItem(strList[6]));
this->setItem(i,7,new QStandardItem(strList[7]));
}
i=i+1;
}
CurrentRow = 0;
qDebug()<<"Data Loading complete!"<<endl;
}
void MTableModel::on_verscrollBar(double value)
{
CurrentRow = value*m_DataList.size();
if(CurrentRow>m_DataList.size()-40)
CurrentRow =m_DataList.size()-40;
updateViewData();
}
int MTableModel::GetCurrentRow()
{
return CurrentRow;
}
void MTableModel::ReceiverChanged(int row)
{
CurrentRow += row;
if(CurrentRow<0||CurrentRow>m_DataList.size()-50)
{
CurrentRow-=row;
return;
}
updateViewData();
}
void MTableModel::updateViewData()
{
this->removeRows(0,this->rowCount());
this->setRowCount(TotalRows);
int k=0;
for(int i=CurrentRow;i<CurrentRow+TotalRows;i++)
{
if(i==m_DataList.size()) break;
QStringList strList = m_DataList[i];
this->setItem(k,0,new QStandardItem(strList[0]));
this->setItem(k,1,new QStandardItem(strList[1]));
this->setItem(k,2,new QStandardItem(strList[2]));
this->setItem(k,3,new QStandardItem(strList[3]));
this->setItem(k,4,new QStandardItem(strList[4]));
this->setItem(k,5,new QStandardItem(strList[5]));
this->setItem(k,6,new QStandardItem(strList[6]));
this->setItem(k,7,new QStandardItem(strList[7]));
k++;
}
qDebug()<<"CurrentRow::"<<CurrentRow<<endl;
}
测试界面:
#include "mainwindow.h"
#include "ui_mainwindow.h"
#include <QFile>
#include <QTextStream>
MainWindow::MainWindow(QWidget *parent) :
QMainWindow(parent),
ui(new Ui::MainWindow)
{
ui->setupUi(this);
this->resize(800,700);
QFile temFile("testdata.txt");
if(temFile.open(QIODevice::WriteOnly))
{
QTextStream out(&temFile);
for(int i = 0;i<1e3;i++)
{
out<<i<<" This sentence is used for testing !"<<endl;
}
}
tableModel = new MTableModel;
tableModel->moveToThread(&m_thread);
m_thread.start();
connect(this,SIGNAL(LoadData(QString)),tableModel,SLOT(LoadData(QString)));
connect(this,SIGNAL(test(int)),tableModel,SLOT(ReceiverChanged(int)));
connect(ui->tableView,SIGNAL(rowchanged(int)),tableModel,SLOT(ReceiverChanged(int)));
connect(ui->tableView,SIGNAL(verticalbarValue(double)),tableModel,SLOT(on_verscrollBar(double)));
ui->tableView->setModel(tableModel);
}
MainWindow::~MainWindow()
{
m_thread.wait();
m_thread.quit();
delete ui;
}
void MainWindow::on_pushButton_clicked()
{
emit LoadData("testdata.txt");
}
实现的效果是,表格数据会根据鼠标滚轮与滑块进行更新。
参考博客:
https://blog.csdn.net/weixin_43676892/article/details/115177208?spm=1001.2014.3001.5506