在我的程序中(使用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?如果是这样,我将如何修复我的代码呢?
答案 0 :(得分:3)
注意moveToThread
必须在对象当前所属的线程上调用,因此您可能需要将对象移回主线程,然后再将其移动到另一个线程。
在顶部的某处添加行mainThread = QtCore.QThread.currentThread()
,并在发出self.moveToThread(mainThread)
信号之前将finished
置于右侧。