QTableVIew加载代理卡顿的解决办法

文章介绍了如何在QTableView中通过设置动态代理来解决大量数据加载导致的卡顿问题,通过分批加载和响应滑动事件实现性能提升。
摘要由CSDN通过智能技术生成

QTableVIew设置代理过程中,由于调用代理非常耗费时间(运算),当数据上千行,加载非常卡顿。

这里介绍动态加载的方法。一开始只设置能显示部分的代理,其余不设置,当滑动条变化的时候,跟着滑动条的变动加载代理,能减少代理的加载时间,实现动态加载代理;

具体就是,先加载10条(可以自定义),刚好将界面覆盖住,当用户往下滑动鼠标滚轮或者滑动滑动条时,向下滑动一个位置就加载一个相应的代理,达到显示的全是代理

演示:

QtableView代码如下:滑动条发生变化时发送信号:

class MyTableView :public QTableView
{
    Q_OBJECT
public:
    explicit MyTableView(QWidget *parent = nullptr);
    void ScrollBarChange(int value);

// protected:
//     void wheelEvent(QWheelEvent *event);

signals:
    void VerticalbarValue(int value);
    // void Rowchanged(int row);
private:
    // 保存当前滑动的最大位置
    int maxArea_ {0};
};

// ---------------------


MyTableView::MyTableView(QWidget *parent) : QTableView(parent)
{
    connect(this->verticalScrollBar(),&QScrollBar::valueChanged, this, &MyTableView::ScrollBarChange);
}

void MyTableView::ScrollBarChange(int value)
{
    qDebug() << " value is:" << value;
    emit VerticalbarValue(value);
    if(value == this->verticalScrollBar()->maximum()) {
        // 滑动到了底部
        qDebug() << "滑动到了底部:  " << value;
        int barSize = this->verticalScrollBar()->size().width();
        emit VerticalbarValue(this->verticalScrollBar()->maximum() + barSize);
    }
}


// void MyTableView::wheelEvent(QWheelEvent *event)
// {
//     if (event->angleDelta().y() > 0) {
//         verticalScrollBar()->setValue(verticalScrollBar()->value() - verticalScrollBar()->singleStep());
//     } else {
//         verticalScrollBar()->setValue(verticalScrollBar()->value() + verticalScrollBar()->singleStep());
//     }

//     QPoint angle = event->angleDelta();
//     if (!angle.isNull()) {
//         int xAngle = angle.x();
//         int yAngle = angle.y();
//         // 使用 xAngle 和 yAngle 处理滚轮事件
//         qDebug() << " x is:" << xAngle << "  y is:" << yAngle;
//     }

//     // emit Rowchanged(row);
// }

显示界面动态加载:

widget.cpp

#include "widget.h"
#include <QVBoxLayout>
#include <QHBoxLayout>
#include <QDebug>

namespace {
const int MAX_COL = 300;
const int MAX_ROW = 3;
const int SHOW_LINE = 10;
}

Widget::Widget(QWidget *parent)
    : QWidget(parent)
{
    this->resize(640,480);
    Init();
    Connect();
}

Widget::~Widget() {}

void Widget::Init()
{
    nameList_<< "Item 1" << "Item 2" << "Item 3" << "Item 4";
    listModel_ = new QStringListModel;
    listModel_->setStringList(nameList_);
    listView_ = new QListView(this);
    listView_->setModel(listModel_);

    // // 创建视图
    tableView_ = new MyTableView(this);
    tableView_->resize(this->size());

    // // 创建模型
    model_ =new QStandardItemModel();
    model_->setHorizontalHeaderLabels({"Column 1", "Column 2", "Column 3"});
    tableView_->setModel(model_);

    QHBoxLayout *hbox = new QHBoxLayout;
    hbox->addWidget(listView_);
    hbox->addWidget(tableView_);
    this->setLayout(hbox);
    UpdateView();
}

void Widget::Connect()
{
    connect(listView_, &QListView::clicked, this ,&Widget::ListViewChange);
    connect(tableView_, &MyTableView::VerticalbarValue, this ,&Widget::DymicDisplay);

    // 快速滑动不丢失代理的关键
    // connect(this, &Widget::FinishAgencySg, tableView_, &MyTableView::AgencyFeedBack);
}

void Widget::UpdateView()
{
    QStandardItem *newItem = nullptr;
    // 设置model, model 目前只设置前十个
    for (int row = 0; row < MAX_COL; row++) {
        for (int col = 0; col < MAX_ROW; ++col){
            newItem = new QStandardItem(QString::number(col));
            model_->setItem(row, col, newItem);
        }
    }

    // 设置代理
    ComboBoxDelegate *comboBoxDelegate = new ComboBoxDelegate();
    tableView_->setItemDelegate(comboBoxDelegate);

    //将前十个数据默认显示,或者全部的数据
    int baseShow = (model_->rowCount() > SHOW_LINE) ? SHOW_LINE : model_->rowCount();
    qDebug() << " max count is :" <<  model_->rowCount() << baseIndex_;
    for (int index = 0; index <= baseShow ; index++) {
        for (int col = 0; col < MAX_ROW; ++col){
            QModelIndex modelIndex = model_->index(index,col, QModelIndex());
            model_->setData(modelIndex, "Option 2"); // 设置初始值为 "Option 2"
            tableView_->openPersistentEditor(model_->index(index,col,QModelIndex()));
        }
    }
    baseIndex_ = 0;
}

void Widget::DymicDisplay(int indexRow)
{
    int max = indexRow - baseIndex_;
    if (max > 0) {
        qDebug() << " baseIndex_ is :" << baseIndex_ << " indexRow is : " << indexRow;
        for (int index = baseIndex_ ; index < indexRow + max; index++) {
            for (int col = 0; col < MAX_ROW; ++col){
                QModelIndex modelIndex = model_->index(index + SHOW_LINE,col, QModelIndex());
                model_->setData(modelIndex, "Option 2"); // 设置初始值为 "Option 2"
                tableView_->openPersistentEditor(model_->index(index + SHOW_LINE,col,QModelIndex()));
            }
        }
        baseIndex_ = indexRow;
    }
}

void Widget::ListViewChange(const QModelIndex &index)
{
    model_->clear();
    UpdateView();
}

widget.h文件:

#ifndef WIDGET_H
#define WIDGET_H

#include "mytableview.h"
#include <QWidget>
#include <QTableView>
#include <QStandardItemModel>
#include <QListView>
#include <QStringListModel>

class Widget : public QWidget
{
    Q_OBJECT

public:
    Widget(QWidget *parent = nullptr);
    ~Widget();

    void Init();
    void Connect();
    void UpdateView();
    void DymicDisplay(int index);

    void ListViewChange(const QModelIndex &index);

private:
    MyTableView *tableView_ {nullptr};
    QStandardItemModel *model_ {nullptr};

    // 列表
    QListView *listView_ {nullptr};
    QStringListModel *listModel_ {nullptr};
    QStringList nameList_ {nullptr};

    int baseIndex_ {0};
};

#endif // WIDGET_H

最后这是代理:

class ComboBoxDelegate : public QItemDelegate {
public:
    ComboBoxDelegate(QObject *parent = nullptr) : QItemDelegate(parent) {}

    // 创建编辑器
    QWidget *createEditor(QWidget *parent, const QStyleOptionViewItem &option, const QModelIndex &index) const override {
        if (!index.isValid())
            return nullptr;

        // 创建一个下拉框
        if (index.column() == 0 || index.column() == 1) {
            QComboBox *comboBox = new QComboBox(parent);
            comboBox->addItem("Option 1");
            comboBox->addItem("Option 2");
            comboBox->addItem("Option 3");
            return comboBox;
        } else if (index.column() == 2) {
            QToolButton * button = new QToolButton(parent);
            button->setIcon(QIcon(":/icon/1.png"));
            return button;
        }

    }

    // 设置编辑器数据
    void setEditorData(QWidget *editor, const QModelIndex &index) const override {
        if (!index.isValid())
            return;

        // 从模型中获取数据并设置给下拉框
        if (index.column() == 0 || index.column() == 1) {
            QString value = index.model()->data(index, Qt::EditRole).toString();
            QComboBox *comboBox = static_cast<QComboBox *>(editor);
            comboBox->setCurrentText(value);
        }
    }

    // 将编辑器的数据写回到模型
    void setModelData(QWidget *editor, QAbstractItemModel *model, const QModelIndex &index) const override {
        if (!index.isValid()) {
            return;
        }
        if (index.column() == 0 || index.column() == 1) {
            QComboBox *comboBox = static_cast<QComboBox *>(editor);
            QString value = comboBox->currentText();
            model->setData(index, value, Qt::EditRole);
        }
    }

    // 更新编辑器的几何属性
    void updateEditorGeometry(QWidget *editor, const QStyleOptionViewItem &option, const QModelIndex &index) const override {
        if (!index.isValid())
            return;
        editor->setGeometry(option.rect);
    }
};
#endif // MYTABLEVIEW_H

  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: QListWidget是Qt中常用的控件之一,用于显示列表。在使用QListWidget时,如果加载的数据过多,会导致界面卡顿。 造成QListWidget卡顿的原因主要有两个方面:一是数据量大,二是界面刷新频繁。 对于数据量大的情况,可以考虑采用分页加载的方式,即每次只加载部分数据,减少一次性加载数据的数量。也可以考虑使用多线程进行数据加载,将数据加载和界面刷新分离,从而减少界面卡顿。 对于界面刷新频繁的情况,可以考虑减少界面刷新的次数。可以将需要刷新的数据先缓存起来,稍后统一进行刷新,从而减少刷新的次数。也可以考虑使用QTableView代替QListWidget,QTableView可以实现按需加载数据,大幅度减少界面卡顿。 总的来说,避免QListWidget加载过多数据卡顿的方法有很多,需要根据具体情况进行选择。关键是充分了解QListWidget的特点和使用方法,做好数据处理和界面优化,可以减少卡顿问题的发生。 ### 回答2: QListWidget是Qt中常用的窗口部件之一,它可以用于显示一个可编辑的列表,支持多种交互方式,如拖放、选择、排序等。但是,当QListWidget中包含过多的数据时,可能会导致卡顿的情况。 QListWidget加载过多卡顿的原因主要有两个:一是数据量过大,二是UI刷新太频繁。 对于数据量过大的问题,可以通过分页处理来解决。当数据过多时,将其分为多页,仅加载当前页的数据,当用户需要查看其他页的数据时再进行加载。这样可以避免QListWidget一次性加载过多的数据造成的卡顿情况。 对于UI刷新太频繁的问题,可以使用QListView来替代QListWidget。QListView是QListWidget的一种更高级别的窗口部件,它可以极大提高GUI的渲染性能。QListView采用Model/View结构,关注数据模型和视图分离。当数据有变化时,只需要更新数据模型,视图会自动进行更新,避免UI刷新过频繁造成的卡顿情况。 除此之外,也可以考虑使用多线程进行数据加载、渲染等操作,从而减少主线程的负担,提高QListWidget的性能表现。 总的来说,对于QListWidget加载过多卡顿的问题,需要根据具体情况采取不同的解决方案。合理地选择数据处理和UI渲染技术,可以达到提高程序性能的目的。 ### 回答3: qlistwidget是一个非常强大的Qt控件,能够轻松地将数据呈现为列表和网格视图。但是,在加载大量数据的情况下,qlistwidget可能会出现卡顿的问题。 造成这个问题的主要原因是由于qlistwidget要在内存中存储所有的数据,如果数据量过大,就会导致CPU和内存的占用率过高,从而造成程序的卡顿和不响应。 解决这个问题的方法有多种,以下是几个可行的方案: 1. 分批加载 将qlistwidget的数据分成几个批次来加载,这样可以减小单次加载的数据量,从而避免在加载大量数据时出现卡顿的情况。 2. 使用QListView或QTableView替代 如果使用qlistwidget的数据仅仅是单纯的显示,可以使用QListView或QTableView替代。这两个控件相比qlistwidget对大量数据的加载更加稳定和流畅,同时还支持分页显示和排序等功能,可以更好地满足用户的需求。 3. 使用Model View架构 采用Model View架构,该架构将数据与视图分离,在内存中只保留需要显示的数据。这种架构可以提高数据加载的效率,同时还可以允许用户对数据进行更加灵活的操作。 4. 可视范围内的数据 只加载可视范围内的数据,超出可视范围的数据暂时不予显示,当用户滚动到可视范围时再进行加载。这种方式可以避免加载大量数据时的卡顿情况。 总之,针对qlistwidget加载过多卡顿的问题,我们可以从分批加载、使用QListView或QTableView、使用Model View架构以及只加载可视范围内的数据等多个角度进行优化,以获得更好的用户体验。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值