Qthread不工作,GUI仍然挂起

时间:2014-05-25 10:28:23

标签: qt pyqt qthread

我正在尝试使用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确实挂断了。

我真的非常感谢任何人的帮助。

1 个答案:

答案 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
...