メインUIをフリーズせずにQStandartItemModelを更新する方法

私はPyQt4を学び始めており、今は何かに長時間拘束されていて、自分自身でそれを理解することはできません:

ここにはコンセプトがあります:カスタムQStandartItemModelを持つTreeViewがあり、数秒ごとに再構築され、たくさんのエントリ(少なくとも何百ものエントリ)を持つことができます。また、異なるカラムなどの代理人も追加されます。デリゲートのない平らなモデルの構築時間は最大3秒になり、TreeViewはフリーズします。

これを解決するための最善の方法についてアドバイスをしてください。どういうわけか、別のスレッドでモデルを構築し、最終的にTreeViewに送信して、新しいモデルでsetModel()を実行するだけですが、その作業を行うことはできませんでした。

ここに問題を少し説明するコードがあります:

from PyQt4.QtCore import *
from PyQt4.QtGui import *
import sys, os, re, time

app = QApplication(sys.argv)
REFRESH = 1

class Reloader_Thread(QThread):
    def __init__(self, parent = None):
        QThread.__init__(self, parent)
        self.loaders = ['\', '--', '|', '/', '--']
        self.emit(SIGNAL('refresh'))
    def run(self):
        format = '|%d/%b/%Y %H:%M:%s| '
        while True:
            self.emit(SIGNAL('refresh'))
            self.sleep(REFRESH)

class Model(QStandardItemModel):
    def __init__(self, viewer=None):
        QStandardItemModel.__init__(self,None)
        self.build()

    def build(self):
        stTime = time.clock()
        newRows = []
        for r in range(1000):
            row = []
            for c in range(12):
                item = QStandardItem('%s %02d%02d' % (time.strftime('%H"%M'%s'), r,c))
                row.append(item)
            newRows.append(row)
        eTime = time.clock() - stTime
        outStr = 'Build %03f' % eTime
        format = '|%d/%b/%Y %H:%M:%s| '
        stTime = time.clock()
        self.beginRemoveRows(QModelIndex(),  0, self.rowCount())
        self.removeRows(0, self.rowCount())
        self.endRemoveRows()
        eTime = time.clock() - stTime
        outStr += ', Remove %03f' % eTime
        stTime = time.clock()
        numNew = len(newRows)
        for r in range(numNew):
            self.appendRow(newRows[r])
        eTime = time.clock() - stTime
        outStr += ', Set %03f' % eTime
        self.emit(SIGNAL('status'), outStr)
        self.reset()

w = QWidget()
w.setGeometry(200,200,800,600)
hb = QVBoxLayout(w)
tv = QTreeView()
tvm = Model(tv)
tv.setModel(tvm)

sb = QStatusBar()
reloader = Reloader_Thread()
tvm.connect(tvm, SIGNAL('status'), sb.showMessage)
reloader.connect(reloader, SIGNAL('refresh'), tvm.build)
reloader.start()

hb.addWidget(tv)
hb.addWidget(sb)
w.show()
app.setStyle('plastique')
app.processEvents(QEventLoop.AllEvents)
app.aboutToQuit.connect(reloader.quit)
app.exec_()

ヒントをありがとう。 これまで私が得た状況は次のとおりです。
私は新しいモデルを構築し、TreeViewに送信します…高速ですが、TreeViewの現在のモデルで何が起きているのか、それをどう対処するのか、私の
‘app’常に増加しています。

もう一つは、私の選択を保持するが、視覚的な矩形や行の順序ではなく、アイテムデータに基づいているので、これもやったことがあるが、適切な方法であるにはあまりにも汚い/ハッキリに見える。これに関する助けもありがとう。コードは次のとおりです。

from PyQt4.QtCore import *
from PyQt4.QtGui import *
import sys, os, re, time

app = QApplication(sys.argv)
REFRESH = 1

class Reloader_Thread(QThread):
    def __init__(self, parent = None):
        QThread.__init__(self, parent)
        self.moveToThread(self)

    def run(self):
        while True:
            model = Model()
            #model.connect(model, SIGNAL('status'), self.emitStat)
            if model.build():
                self.emit(SIGNAL('refresh'), model)
            self.sleep(REFRESH)

    def emitStat(self, stat):
        self.emit(SIGNAL('status'), stat)

class Tree(QTreeView):
    def __init__(self, parent=None):
        QTreeView.__init__(self, parent)
        self.setSelectionMode(QAbstractItemView.ExtendedSelection)

    def resetModel(self, model):
        stTime = time.clock()
        # gather old selection
        oldSel = set()
        currModel = self.model()
        for index in self.selectedIndexes():
            id = currModel.itemFromIndex(index).data().toString()
            oldSel.add('%s'%id)
        # setup new
        self.setModel(model)
        selModel = self.selectionModel()
        for r in range(model.rowCount()):
            item = model.item(r,0)
            rowId = '%s' % item.data().toString()
            if rowId in oldSel:
                sel = QItemSelection(model.index(r,0), model.index(r,model.columnCount()-1))
                selModel.select(sel, QItemSelectionModel.Select)
        self.setSelectionModel(selModel)
        self.emit(SIGNAL('status'), 'TV setModel: %03fs' % (time.clock() - stTime))

class Model(QStandardItemModel):
    def __init__(self, viewer=None):
        QStandardItemModel.__init__(self,None)

    def build(self):
        stTime = time.clock()
        newRows = []
        for r in range(1000):
            row = []
            var = QVariant('%d'%r)
            for c in range(12):
                item = QStandardItem('%s r%02dc%02d' % (time.strftime('%H"%M'%s'), r,c))
                item.setData(var)
                row.append(item)
            newRows.append(row)
        eTime = time.clock() - stTime
        outStr = 'Build %03f' % eTime
        format = '|%d/%b/%Y %H:%M:%s| '
        stTime = time.clock()
        self.beginRemoveRows(QModelIndex(),  0, self.rowCount())
        self.removeRows(0, self.rowCount())
        self.endRemoveRows()
        eTime = time.clock() - stTime
        outStr += ', Remove %03f' % eTime
        stTime = time.clock()
        numNew = len(newRows)
        for r in range(numNew):
            self.appendRow(newRows[r])
        eTime = time.clock() - stTime
        outStr += ', Set %03f' % eTime
        self.emit(SIGNAL('status'), outStr)
        #self.reset()
        return True

w = QWidget()
w.setGeometry(200,200,800,600)
hb = QVBoxLayout(w)
tv = Tree()

sb = QStatusBar()
reloader = Reloader_Thread()
tv.connect(tv, SIGNAL('status'), sb.showMessage)
reloader.connect(reloader, SIGNAL('refresh'), tv.resetModel)
reloader.connect(reloader, SIGNAL('status'), sb.showMessage)
reloader.start()

hb.addWidget(tv)
hb.addWidget(sb)
w.show()
app.setStyle('plastique')
app.processEvents(QEventLoop.AllEvents)
app.aboutToQuit.connect(reloader.quit)
app.exec_()
ベストアンサー

あなたは正しい考えを持っています。

モデルを再計算するには、別々のワーカースレッドが必要です(必要に応じて、時間ベースまたは信号ベースで)。次に、メインスレッドに通知するために計算が完了したら、シグナルをフックアップする必要があります。

あなたが知っておかなければならない問題がいくつかあります。

  1. QObjectはスレッド内に存在します。シグナル/スロットのパラダイム(少なくともC ++
    QTでは)は、デフォルトでオーナースレッドにローカルで動作します。シグナルクロススレッドを送信する場合は、明示的に指定する必要があります(signal/connectのドキュメントを参照)。

  2. >

  3. メインスレッドとワーカースレッドが適切に同期されていることを確認します。

QTには、メインスレッドで新しいモデルを保持するために使用できるQFuture(PyQTがあればわからない)があり、ワーカースレッドがそれを再生成するときに自動的にリロードすることができます。

がんばろう。

コメントする

メールアドレスが公開されることはありません。 * が付いている欄は必須項目です