如何从线程内访问外部属性

Posted

技术标签:

【中文标题】如何从线程内访问外部属性【英文标题】:How can I access an outside attribute from within a Thread 【发布时间】:2018-04-16 05:16:24 【问题描述】:

我用 PyQt5 Designer 设计了一个 GUI。

我希望 tryingMethod 始终检查来自服务器的传入消息,所以我创建了一个线程,以便 GUI 和 run 方法可以同时运行,这很有效。

问题是,当我尝试从 Ui_Form 访问变量“Button”时,会引发以下错误:AttributeError: 'Ui_Form' object has no attribute 'Button'。

我尝试将 self 作为tryMethod 线程的参数传递给线程,但它给了我“意外类型:(() -> None, Ui_Form) 警告”

我尝试创建几个不同的类,但我无法解决问题。任何帮助表示赞赏! :)

class Ui_Form(object):
    def __init__(self):
        t = threading.Thread(target=self.tryingMethod)
        t.start()




    def tryingMethod(self):
        self.Button.setText("TESTING")  ##This doesn't work.

        while True:
            message = self.clientSocket.receive()



    def setupUi(self, Form):
        //Code has been shortened
        self.Button.setFont(font)
        self.Button.setCursor(QtGui.QCursor(QtCore.Qt.ForbiddenCursor))
        self.Button.setObjectName("Button")



if __name__ == "__main__":
     app = QtWidgets.QApplication(sys.argv)
     Form = QtWidgets.QWidget()
     ui = Ui_Form()
     ui.setupUi(Form)
     Form.show()
     sys.exit(app.exec_())

【问题讨论】:

【参考方案1】:

Qt 不允许直接从另一个线程更新 GUI,一个可能的解决方案是使用信号,但为此我们需要一个继承 QObject 的类,另一方面不建议修改生成的类通过 Qt Designer,该类仅用于填充小部件因此您可以利用创建一个从小部件继承的类,并且由于它是一个小部件也继承自 QObject,因此我们将在该类中创建一个信号,并将该信号连接到按钮的setText() 方法。

class Ui_Form(object):
    def setupUi(self, Form):
        # Code has been shortened
        self.Button.setFont(font)
        self.Button.setCursor(QtGui.QCursor(QtCore.Qt.ForbiddenCursor))
        self.Button.setObjectName("Button")

class Form(QtWidgets.QWidget, Ui_Form):
    someSignal = QtCore.pyqtSignal(str)
    def __init__(self, *args, **kwargs):
        QtWidgets.QWidget.__init__(self, *args, **kwargs)
        self.setupUi(self)
        t = threading.Thread(target=self.tryingMethod)
        self.someSignal.connect(self.Button.setText)
        t.start()

    def tryingMethod(self):
        self.someSignal.emit("TESTING")  ##This doesn't work.
        while True:
            message = self.clientSocket.receive()


if __name__ == "__main__":
     app = QtWidgets.QApplication(sys.argv)
     w = Form()
     w.show()
     sys.exit(app.exec_())

【讨论】:

【参考方案2】:

在调用 ui.setupUi(Form) 之前,您正在实例化 ui = Ui_Form()

ui = Ui_Form() 接下来将调用__init__ 方法,该方法依次开始创建并启动您的线程。这将(在最坏的情况下)尝试在按钮创建之前设置它,这可能是您收到错误的原因。

您需要先创建按钮,然后再尝试使用它。一种选择是将线程创建移到__init__ 之外,如下所示:

class Ui_Form(object):
    def startThread(self):
        t = threading.Thread(target=self.tryingMethod)
        t.start()

    def tryingMethod(self):
        self.Button.setText("TESTING")  ##This doesn't work.
        while True:
            message = self.clientSocket.receive()

    def setupUi(self, Form):
        # Code has been shortened
        self.Button.setFont(font)
        self.Button.setCursor(QtGui.QCursor(QtCore.Qt.ForbiddenCursor))
        self.Button.setObjectName("Button")


if __name__ == "__main__":
     app = QtWidgets.QApplication(sys.argv)
     Form = QtWidgets.QWidget()
     ui = Ui_Form()
     ui.setupUi(Form)
     Form.show()

     # call after creation
     ui.startThread()

     sys.exit(app.exec_())

也就是说,您的代码还有一些其他错误,例如提到的 eyllanesc。

【讨论】:

以上是关于如何从线程内访问外部属性的主要内容,如果未能解决你的问题,请参考以下文章

如何获取组件数据?如何从外部访问组件?

如何使用 Angular JS 从外部范围访问数组值

内部类

从 MapView 组件访问 ref 属性以供外部函数使用

java线程在创建外部对象之前访问它

为啥内部类的private变量可被外部类直接访问