Python PyQt:是否可以将QThread与非GUI程序一起使用?

时间:2014-08-19 20:32:13

标签: python multithreading pyqt pyqt4

我有一个Python PyQt应用程序,它显示一个简单的UI。 当用户单击UI中的按钮时,它会触发QThread。 线程的使用阻止了用户界面冻结"而线程运行。 我发出信号以将信息从运行线程传递回UI以进行状态更新并指示完成。一切正常如上所述,我为我的UI创建了一个简单的类来调用它创建线程并运行我的通用处理。

但是,我还想创建我的程序的命令行版本(No GUI)并使用相同的处理QThread类。但是,当我尝试连接信号时,出现以下错误。 QThread似乎只适用于GUI程序?

AttributeError: MyClass instance has no attribute 'connect'

是否可以将QThread与非GUI程序一起使用?

from PyQt4 import QtCore
from PyQt4.QtCore import * 

#======================================

class MyProcess(QThread):

    def __init__(self):
        QThread.__init__(self)

    def __del__(self):
        self.quit()
        self.wait()  

    def run(self):
        print "do time intensive process here"  
        self.emit( SIGNAL('processdone'), "emitting signal processdone") 
        return       

#====================================== 

class MyClass(QObject):

    def __init__(self, parent=None):            # All QObjects receive a parent argument (default to None)
        super(MyClass, self).__init__(parent)   # Call parent initializer.

        thread1 = MyProcess()  # uses QThread and emits signal 'processdone' 
        self.connect( thread1, SIGNAL("processdone"), self.thread1done)    
        thread1.start()  

    def thread1done(self):
        print "done"      

#======================================

if __name__ == "__main__": 

    MyClass()

1 个答案:

答案 0 :(得分:5)

问题不在于QThread,问题是您从没有使用它的类中调用connect方法。您需要使MyClass继承自QObject。

在GUI中这是有效的,因为你正在使用的任何小部件(QDialog,QMainWindow,QWidget ......)它(直接或间接)从QObject继承。

要使MyClass继承自QObject,您只需:

class MyClass(QObject):                         # Specify the class your are specializing.
    def __init__(self, parent=None):            # All QObjects receive a parent argument (default to None)
        super(MyClass, self).__init__(parent)   # Call parent initializer.

        # And countinue your code here... 

我还建议您使用New-style Signal and Slot Support

除了调用processdone信号外,一切正常,但显然它永远不会触发对thread1done的调用。

我能发现的问题是你没有定义一个处理过的信号。 Qt不存在该信号。检查我离开的链接,了解自定义信号。同时你可以添加:

class MyProcess(QThread):
    processdone = QtCore.pyqtSignal("QString")

在课程开始时。

最后一件事,但非常重要。您没有使用GUI,但您仍在使用QObjects和Qt信号机制,这取决于主要的Qt循环。因此,尽管你的应用程序是非gui程序,你仍然需要一个QApplication对象。

这是您的代码,现在正在运行:

from PyQt4 import QtCore
from PyQt4 import QtGui
from PyQt4.QtCore import * 

class MyProcess(QThread):
    processdone = QtCore.pyqtSignal("QString") # Define custom signal.
    def __init__(self, parent = None):
        QThread.__init__(self, parent)
    def run(self):
        print("do time intensive process here")
        self.emit( SIGNAL('processdone'), "emitting signal processdone")
        return       

class MyClass(QObject):

    def __init__(self, parent=None):            # All QObjects receive a parent argument (default to None)
        super(MyClass, self).__init__(parent)   # Call parent initializer.

        thread1 = MyProcess(self) 
        self.connect( thread1, SIGNAL("processdone"), self.thread1done)    
        thread1.start()  

    @QtCore.pyqtSlot("QString")         # Tell Python this is a QTSLOT an receives a string
    def thread1done(self, text):
        print(text)                     # Print the text from the signal.

if __name__ == "__main__":
    import sys
    app = QtGui.QApplication(sys.argv)  # You still need a QApplication object.
    a = MyClass()
    sys.exit(app.exec())