1

いくつかのブロッキング コードを別の QThread に移動することで、PyQt4 プログラムの応答性を高めようとしました。うまくいかなかったので、デモンストレーションとしてこの最小限の例を作成しました。

import sys
import time
from PyQt4 import QtCore, QtGui


class Sleeper(QtCore.QObject):

    def __init__(self):
        super(Sleeper, self).__init__()
        self.timer = QtCore.QTimer()
        self.timer.timeout.connect(self.sleep)
        self.timer.start(1000)

    def sleep(self):
        time.sleep(1)


class MyGUI(QtGui.QMainWindow):

    def __init__(self, parent=None):
        super(MyGUI, self).__init__(parent)
        self.button = QtGui.QPushButton("hello", self)
        self.sleeper = Sleeper()
        self.thread = QtCore.QThread()
        self.sleeper.moveToThread(self.thread)
        self.thread.start()


if __name__ == "__main__":
    qapp = QtGui.QApplication(sys.argv)
    my_gui = MyGUI()
    my_gui.show()
    qapp.exec_()

このコードの問題は、sleep-command がまだユーザー インターフェイスをブロックしていることです。Sleeper クラスの外部で QTimer を作成、接続、実行すると期待どおりに動作することがわかりましたが、その理由はわかりません。

4

1 に答える 1

5

connect、オブジェクトが GUI スレッドにある間に呼び出されるため、イベント ハンドラーも GUI スレッドで実行されます。最初にスレッドに移動してから、接続を作成してみてください。

class Sleeper(QtCore.QObject):

    def __init__(self):
        super(Sleeper, self).__init__()
        self.timer = QtCore.QTimer()

    def initialize(self): # Creating the connection and starting separately
        self.timer.timeout.connect(self.sleep)
        self.timer.start(1000)

    def sleep(self):
        time.sleep(1)

class MyGUI(QtGui.QMainWindow):

    def __init__(self, parent=None):
        super(MyGUI, self).__init__(parent)
        self.button = QtGui.QPushButton("hello", self)
        self.sleeper = Sleeper()
        self.thread = QtCore.QThread()
        self.sleeper.moveToThread(self.thread) # Move to thread
        self.sleeper.initialize() # Create connection now
        self.thread.start()

また、これを確認してください: スレッド間の Qt 接続タイプ: なぜこれが機能するのですか?

于 2014-09-23T12:55:56.967 に答える