我正在尝试使用pyqt实现线程的基本示例,其中有一个文本框在处理某些代码时会定期更新。我试图删除任何不必要的依赖并尽可能地抽象代码,所以我只是在后台用一个while循环替换要处理的代码,该循环在每个循环中发送数据以供文本框读取。然而,它不起作用,我不知道是什么导致UI挂断。我为当前线程id添加了调试语句,main和worker类(在我的例子中称为Process)匹配
#!/zin/tools/bin/python
#vim:expandtab filetype=python nocindent sw=4
import sys, traceback
import os
import subprocess
import time
from PyQt4 import QtCore,QtGui
try:
_fromUtf8 = QtCore.QString.fromUtf8
except AttributeError:
_fromUtf8 = lambda s: s
class Process (QtCore.QObject):
processCmdDone = QtCore.pyqtSignal()
processdataReady = QtCore.pyqtSignal(str)
def __init__(self):
super(Process, self).__init__()
self.processdataReady.connect(self.debug)
def debug(self):
print "signal process data ready is invoked from within the process"
@QtCore.pyqtSlot()
@QtCore.pyqtSlot(str)
def execCmd(self):
print 'worker thread id :'+ str (QtCore.QThread.currentThreadId())
x = 0
while x < 100:
x+=1
self.processdataReady.emit(str(x)+'\n')
time.sleep(1)
self.processCmdDone.emit()
print "process ended"
class MainWindow (QtGui.QWidget):
def __init__(self,filename = None, parent=None):
super(MainWindow,self).__init__(parent)
self.transcript_textEdit = QtGui.QTextBrowser()
font = QtGui.QFont()
font.setPointSize(12)
self.transcript_textEdit.setFont(font)
self.transcript_textEdit.setObjectName(_fromUtf8("transcript_textEdit"))
self.layout = QtGui.QVBoxLayout(self)
self.layout.addWidget(self.transcript_textEdit)
self.run_pushButton = QtGui.QPushButton()
self.run_pushButton.setFont(font)
self.run_pushButton.setObjectName(_fromUtf8("run_pushButton"))
self.run_pushButton.setText('Run')
self.layout.addWidget(self.run_pushButton)
self.run_pushButton.clicked.connect(self.execCmdThreading)
def dataReady(self,text):
cursor = self.transcript_textEdit.textCursor()
cursor.movePosition(cursor.End)
cursor.insertText(str(text))
self.transcript_textEdit.ensureCursorVisible()
def execCmdThreading(self):
print 'gui thread id :'+ str (QtCore.QThread.currentThreadId())
thread = QtCore.QThread(self)
process_inst = Process()
process_inst.moveToThread(thread)
process_inst.processdataReady.connect(self.dataReady)
process_inst.processCmdDone.connect(thread.quit)
thread.finished.connect(thread.deleteLater)
process_inst.processCmdDone.connect(process_inst.deleteLater)
thread.started.connect(lambda: process_inst.execCmd())
thread.start()
if __name__=="__main__":
import sys
app = QtGui.QApplication(sys.argv)
MainWindowInst = MainWindow()
MainWindowInst.show()
sys.exit(app.exec_())
我只是创建一个文本框和一个按钮,在单击按钮时,为类'Process'的inst创建一个线程,其中moveToThread和它的方法'execCmd'被执行。在执行execCmd方法时,会发出一个信号'processdataReady',该信号有望在创建的文本框中显示文本。但是,代码似乎没有按预期运行,并且UI确实挂断了。
我真的非常感谢任何人的帮助。
答案 0 :(得分:1)
您正在将thread.started
信号连接到lambda:
thread.started.connect(lambda: process_inst.execCmd())
在这里使用正则python可调用将始终导致在gui线程中处理信号。这是可以理解的,因为可调用的(lambda)没有QObjects所具有的线程亲和性。从lambda中,process_inst.execCmd
方法然后同步执行,对象具有什么线程亲和性并不重要。因此GUI线程将阻止。
如果您希望在工作线程事件lopp中接收和处理信号,请将其直接连接到插槽。
为此,您还需要确保保留对process_inst
的引用,否则当它超出范围时将被销毁。
通过这些调整,您的程序适合我:
...
thread.started.connect(process_inst.execCmd)
thread.process_inst = process_inst
...