如何在Pyqt中的QThread之间来回移动对象

时间:2013-05-11 16:22:28

标签: python pyqt qthread

在我的程序中(使用Python 2.7),我创建了一个包含一些重要数据和方法的对象。有些方法是CPU饥饿的,所以在某些情况下我会在CPU密集型方法的持续时间内将对象移动到新的QThread,然后让它们返回到主线程。稍后,当调用CPU密集型方法时,我想再次将对象移动到另一个QThread,但是这会失败,说“当前线程不是对象的线程”。

这是一个重现问题的简单例子:

import sys
from PyQt4 import QtCore, QtGui
from time import sleep

class ClassA(QtGui.QDialog):
    def __init__(self):
        super(ClassA, self).__init__()
        mainLayout=QtGui.QVBoxLayout()
        self.lineEdit=QtGui.QLineEdit()
        mainLayout.addWidget(self.lineEdit)
        self.setLayout(mainLayout)
        self.show()
        self.obj=ClassC(self)        
        self.executeProgram()
    def executeProgram(self):
        self.lineEdit.setText("Starting new thread...")
        self.thread=QtCore.QThread()
        self.obj.moveToThread(self.thread)        
        self.thread.started.connect(self.obj.doWork)
        self.obj.doingWork.connect(self.updateGui)
        self.obj.finished.connect(self.killThread)
        self.thread.start()
    def updateGui(self,message):
        self.lineEdit.setText(message)
    def killThread(self):
        self.thread.quit()
        self.thread.wait()
        self.obj.finished.disconnect()
        self.executeProgram()

class ClassC(QtCore.QObject):
    finished=QtCore.pyqtSignal()
    doingWork=QtCore.pyqtSignal(str)
    def __init__(self,parent=None):
            super(ClassC, self).__init__()
    def doWork(self):
        for i in range(5):
            self.doingWork.emit("doing work: iteration "+str(i))
            sleep(1)
        self.finished.emit()

if __name__=="__main__":
    app=QtGui.QApplication(sys.argv)
    obj=ClassA()
    app.exec_()       

是否可以多次将对象移动到其他QThread?如果是这样,我将如何修复我的代码呢?

1 个答案:

答案 0 :(得分:3)

注意moveToThread必须在对象当前所属的线程上调用,因此您可能需要将对象移回主线程,然后再将其移动到另一个线程。

在顶部的某处添加行mainThread = QtCore.QThread.currentThread(),并在发出self.moveToThread(mainThread)信号之前将finished置于右侧。