QHeaderView::ResizeToContent在QTableView中垂直标题的性能问题

Performance issue with QHeaderView::ResizeToContents for vertical header in QTableView

本文关键字:标题 问题 垂直 性能 ResizeToContent QTableView QHeaderView      更新时间:2023-10-16

给定一个显示具有单列但数百行的自定义项目模型的QTableView,将垂直标题的sectionResizeMode设置为ResizeToContents会对性能产生巨大的负面影响。

出于调试原因,我在项模型的::data方法中添加了一些输出,以查看视图实际查询了哪些行。事实证明,表视图实际上会在需要在将调整大小模式设置为ResizeToContents的情况下进行渲染时立即查询模型中的每一行 - 无论显示多少行。

下面的代码示例给出了以下输出:

*** show ***
query  0
query  1
query  2
query  3
... many lines trimmed ...
query  495
query  496
query  497
query  498
query  499
query  0
query  1
query  2
query  3
... many lines trimmed ...
query  495
query  496
query  497
query  498
query  499
query  0
query  1
query  2
query  3
query  4
query  5
query  6
query  0
query  1
query  2
query  3
query  4
query  5
query  6

也就是说,视图首先似乎对所有行迭代两次。然后,它会循环访问在表视图的视口中实际可见的行。因为它恰好在我的屏幕上,有七行可见。

注释掉感兴趣线后,示例的输出减少为:

*** show ***
query  0
query  1
query  2
query  3
query  0
query  1
query  2
query  3

由于这些行现在的默认高度比以前略大,因此现在只有四行可见。更重要的是,现在总共只从模型中查询了八行。

为什么会有这种奇怪的行为?

SCCE

scce.pro

QT += core gui widgets
CONFIG += c++11
TARGET = sscce
TEMPLATE = app
SOURCES += main.cpp

main.cc

#include <QAbstractItemModel>
#include <QApplication>
#include <QHeaderView>
#include <QTableView>
#include <QDebug>
class Model: public QAbstractItemModel {
public:
int rowCount(const QModelIndex &parent) const {
return parent.isValid() ? 0 : 500;
}
int columnCount(const QModelIndex &parent) const {
return parent.isValid() ? 0 : 1;
}
QModelIndex parent(const QModelIndex &/* child */) const {
return QModelIndex();
}
QModelIndex index(int row, int column, const QModelIndex &parent) const {
return parent.isValid() ? QModelIndex() : createIndex(row, column, Q_NULLPTR);
}
QVariant data(const QModelIndex &index, int role) const {
if (role == Qt::DisplayRole)
qDebug() << "query " << index.row();
return (index.isValid() && (role == Qt::DisplayRole)) ?
QStringLiteral("Row %1").arg(index.row()) : QVariant();
}
QVariant headerData(int section, Qt::Orientation orientation, int role) const {
return ((orientation == Qt::Vertical) && (role == Qt::DisplayRole)) ? section : QVariant();
}
};
int main(int argc, char *argv[]) {
QApplication app(argc, argv);
QTableView view;
view.setModel(new Model());
/* Line of interest: */
view.verticalHeader()->setSectionResizeMode(QHeaderView::ResizeToContents);
qDebug() << "*** show ***";
view.show();
return app.exec();
}
ResizeToContents

为所有行设置相同的高度,甚至更多,它设置最适合所有行的高度,因此它必须查询整个表。

如果行是常量,并且您可以手动确定其高度,则可以尝试存储所需的最小高度,然后安装事件过滤器以调整大小并手动设置高度。其他方案类似,但处理起来更复杂(在添加/删除/编辑行等时必须连接到信号)。