Медленный выбор в QTreeView, почему?

Недавно я столкнулся со стеной в проекте, над которым я работаю и который использует PyQt. У меня есть QTreeView, подключенный к QAbstractItemModel, который обычно имеет тысячи узлов. Пока все работает нормально, но сегодня я понял, что выбор большого количества узлов очень медленный. После некоторого копания оказалось, что QAbstractItemModel.parent() вызывается слишком часто. Я создал минимальный код для воспроизведения проблемы:

#!/usr/bin/env python
import sys
import cProfile
import pstats

from PyQt4.QtCore import Qt, QAbstractItemModel, QVariant, QModelIndex
from PyQt4.QtGui import QApplication, QTreeView

# 200 root nodes with 10 subnodes each

class TreeNode(object):
    def __init__(self, parent, row, text):
        self.parent = parent
        self.row = row
        self.text = text
        if parent is None: # root node, create subnodes
            self.children = [TreeNode(self, i, unicode(i)) for i in range(10)]
        else:
            self.children = []

class TreeModel(QAbstractItemModel):
    def __init__(self):
        QAbstractItemModel.__init__(self)
        self.nodes = [TreeNode(None, i, unicode(i)) for i in range(200)]

    def index(self, row, column, parent):
        if not self.nodes:
            return QModelIndex()
        if not parent.isValid():
            return self.createIndex(row, column, self.nodes[row])
        node = parent.internalPointer()
        return self.createIndex(row, column, node.children[row])

    def parent(self, index):
        if not index.isValid():
            return QModelIndex()
        node = index.internalPointer()
        if node.parent is None:
            return QModelIndex()
        else:
            return self.createIndex(node.parent.row, 0, node.parent)

    def columnCount(self, parent):
        return 1

    def rowCount(self, parent):
        if not parent.isValid():
            return len(self.nodes)
        node = parent.internalPointer()
        return len(node.children)

    def data(self, index, role):
        if not index.isValid():
            return QVariant()
        node = index.internalPointer()
        if role == Qt.DisplayRole:
            return QVariant(node.text)
        return QVariant()


app = QApplication(sys.argv)
treemodel = TreeModel()
treeview = QTreeView()
treeview.setSelectionMode(QTreeView.ExtendedSelection)
treeview.setSelectionBehavior(QTreeView.SelectRows)
treeview.setModel(treemodel)
treeview.expandAll()
treeview.show()
cProfile.run('app.exec_()', 'profdata')
p = pstats.Stats('profdata')
p.sort_stats('time').print_stats()

Чтобы воспроизвести проблему, просто запустите код (который выполняет профилирование) и выберите все узлы в виджете дерева (либо с помощью выбора сдвига, либо с помощью Cmd-A). Когда вы выйдете из приложения, статистика профилирования покажет что-то вроде:

Fri May  8 20:04:26 2009    profdata

         628377 function calls in 6.210 CPU seconds

   Ordered by: internal time

   ncalls  tottime  percall  cumtime  percall filename:lineno(function)
        1    4.788    4.788    6.210    6.210 {built-in method exec_}
   136585    0.861    0.000    1.182    0.000 /Users/hsoft/Desktop/slow_selection.py:34(parent)
   142123    0.217    0.000    0.217    0.000 {built-in method createIndex}
    17519    0.148    0.000    0.164    0.000 /Users/hsoft/Desktop/slow_selection.py:52(data)
   162198    0.094    0.000    0.094    0.000 {built-in method isValid}
     8000    0.055    0.000    0.076    0.000 /Users/hsoft/Desktop/slow_selection.py:26(index)
   161357    0.047    0.000    0.047    0.000 {built-in method internalPointer}
       94    0.000    0.000    0.000    0.000 /Users/hsoft/Desktop/slow_selection.py:46(rowCount)
      404    0.000    0.000    0.000    0.000 /Users/hsoft/Desktop/slow_selection.py:43(columnCount)
       94    0.000    0.000    0.000    0.000 {len}
        1    0.000    0.000    6.210    6.210 <string>:1(<module>)
        1    0.000    0.000    0.000    0.000 {method 'disable' of '_lsprof.Profiler' objects}

Самое странное в этих данных то, как часто вызывается parent(): 136 тысяч раз для 2 тысяч узлов! Кто-нибудь знает, почему?


person Virgil Dupras    schedule 08.05.2009    source источник


Ответы (1)


Попробуйте вызвать setUniformRowHeights(true) для просмотра дерева:

https://doc.qt.io/qt-4.8/qtreeview.html#uniformRowHeights-prop

Кроме того, есть инструмент C++ под названием modeltest от qt labs. Я не уверен, что есть что-то для python:

https://wiki.qt.io/Model_Test

person Mark Beckwith    schedule 10.05.2009
comment
Спасибо за подсказку, но, к сожалению, она не помогла. Это уменьшило количество родительских звонков, но только до 134 тысяч звонков. Что касается Modeltest, это кажется интересным, но я не знаю, как импортировать сторонние компоненты C++ в PyQt (придется погуглить). Но в любом случае мне кажется, что эта модель правильная, не так ли? - person Virgil Dupras; 10.05.2009