QTeaderView中的垂直标头的QHeaderView :: ResizeToContents性能问题

时间:2016-12-29 16:44:43

标签: c++ qt

如果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

由于行现在的默认高度比之前略大,现在只能看到四行。更重要的是,现在总共只能从模型中查询8行。

为什么会出现这种奇怪的行为?

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();
}

1 个答案:

答案 0 :(得分:0)

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

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