QTableView+自定义model实现大量数据的不卡顿刷新

QTableView在显示大量数据时,即使在子线程中对数据模型进行数据更改,依然会导致界面的卡顿,而且数据量较大时,会占用大量的内存,为解决此问题,可以在建立自定义的数据模型,替代QStandardItemModel,并在数据模型中存储需要显示的数据,但是在界面中只显示部分数据,同时监测用户操作,当需要刷新数据时,对数据进行重新加载。
核心思路即是:由于QTableView 一次能够显示的数据有限,所以只在QTableView显示部分数据,当用户操作时,更新数据。
需要用到的知识:

  1. 多线程;
  2. 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

  • 10
    点赞
  • 120
    收藏
    觉得还不错? 一键收藏
  • 6
    评论
实现时间策略表格,你可以使用 QTableView自定义委托来实现。 首先,你需要创建一个 QTableView 实例,并将其设置为可编辑。然后,你需要创建一个自定义委托来处理表格中单元格的显示和编辑。 下面是一个简单的示例代码,演示如何创建一个时间策略表格: ```python from PyQt5.QtWidgets import QApplication, QTableView, QAbstractItemView, QHeaderView from PyQt5.QtGui import QStandardItemModel, QStandardItem from PyQt5.QtCore import Qt, QTime class TimePolicyDelegate(QtWidgets.QStyledItemDelegate): def createEditor(self, parent, option, index): editor = QtWidgets.QTimeEdit(parent) editor.setDisplayFormat("hh:mm") return editor def setEditorData(self, editor, index): value = index.model().data(index, Qt.EditRole) time = QTime.fromString(value, "hh:mm") editor.setTime(time) def setModelData(self, editor, model, index): time = editor.time() model.setData(index, time.toString("hh:mm"), Qt.EditRole) class TimePolicyTableView(QTableView): def __init__(self, parent=None): super().__init__(parent) self.setEditTriggers(QAbstractItemView.DoubleClicked) self.setSelectionBehavior(QAbstractItemView.SelectRows) self.horizontalHeader().setSectionResizeMode(QHeaderView.Stretch) self.verticalHeader().setVisible(False) self.setItemDelegate(TimePolicyDelegate(self)) self.setModel(QStandardItemModel()) self.model().setColumnCount(2) self.model().setHeaderData(0, Qt.Horizontal, "时间段") self.model().setHeaderData(1, Qt.Horizontal, "策略") def addTimePolicy(self, time, policy): row = self.model().rowCount() self.model().setItem(row, 0, QStandardItem(time.toString("hh:mm"))) self.model().setItem(row, 1, QStandardItem(policy)) ``` 这个示例代码创建了一个名为 TimePolicyTableView 的自定义 QTableView 子类。它使用 QStandardItemModel 来存储时间策略表格中的数据,并使用 QHeaderView 和 QAbstractItemView 来设置表格的外观和行为。 自定义委托 TimePolicyDelegate 负责将时间数据显示为 QTimeEdit 控件,并将其转换回字符串格式,以便在模型中存储。 你可以使用 addTimePolicy 方法来向表格中添加时间策略。该方法将时间和策略作为参数,并将其插入到模型的新行中。 最后,你可以在主函数中创建 TimePolicyTableView 的实例,并将其添加到应用程序中。例如: ```python if __name__ == "__main__": app = QApplication(sys.argv) table = TimePolicyTableView() table.addTimePolicy(QTime(8, 0), "开门") table.addTimePolicy(QTime(12, 0), "关门") table.addTimePolicy(QTime(13, 0), "开门") table.addTimePolicy(QTime(17, 0), "关门") table.show() sys.exit(app.exec_()) ``` 这将创建一个简单的时间策略表格,其中包含四个时间段和相应的策略。
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值