PyQt:如何从 QThread 获取 UI 数据

Posted

技术标签:

【中文标题】PyQt:如何从 QThread 获取 UI 数据【英文标题】:PyQt: How to get UI data from a QThread 【发布时间】:2017-07-01 05:55:54 【问题描述】:

我有以下代码,但它抱怨我无法从我的线程访问 UI 数据。在下面的示例代码中,我可以访问userInputString 值以便我的线程可以运行的最佳方式是什么?

self.nameField 是 PyQt QLineEdit。

QObject::setParent: 无法设置父级,新父级在不同的线程中 QPixmap:在 GUI 线程之外使用像素图是不安全的 QWidget::repaint:检测到递归重绘

import myUI

class MainUIClass(QtGui.QMainWindow, myUI.Ui_MainWindow):
    def __init__(self, parent=None):
        super(MainUIClass, self).__init__(parent)
        self.setupUi(self)

        self.startbutton.clicked.connect(self.do_work)

        self.workerThread = WorkerThread()
        self.connect(self.workerThread, SIGNAL("myThreading()"), self.myThreading, Qt.DirectConnection)

    def do_work(self):
        self.userInputString = self.nameField.Text()
        self.workerThread.start()

    def myThreading(self):

        if userInputString is not None:
            #Do something

class WorkerThread(QThread):
    def __init__(self, parent=None):
        super(WorkerThread, self).__init__(parent)

    def run(self):
        self.emit(SIGNAL("myThreading()"))

if __name__ == '__main__':
    a = QtGui.QApplication(sys.argv)
    app = MainUIClass()
    app.show()
    a.exec_()

【问题讨论】:

您应该阅读“如何以正确的方式使用 QThread”***.com/questions/16879971/…blog.debao.me/2013/08/… 您也在使用旧式信号...对于您的问题,您应该使用信号和插槽...你在一个线程中运行你的进程,然后从这个线程向可以访问 GUI 元素的 GUI 线程发送信号 感谢您指出这一点。我要做的就是能够从我的线程类中访问“userInputString”值,但不知道如何将该值从 MainUI 类传递给线程类。您认为共享多线程 Array 或 Value 数据类型会是另一种实现方式吗? @joke4me。您问题中的代码不会产生这些错误消息。事实上,稍作修正后,它的工作原理完全符合预期。请阅读有关如何提供minimal reproducible example 的指南。 【参考方案1】:

不确定它是否是您需要的,但这是一个使用 Qt5 的工作 QThread 示例

import time
import sys
from PyQt5 import QtWidgets, QtGui, QtCore

class MainWindow(QtWidgets.QMainWindow):
    def __init__(self, parent=None):
        super(MainWindow, self).__init__(parent)
        self.worker_thread = WorkerThread()
        self.worker_thread.job_done.connect(self.on_job_done)
        self.create_ui()

    def create_ui(self):
        self.button = QtWidgets.QPushButton('Test', self)
        self.button.clicked.connect(self.start_thread)
        layout = QtWidgets.QVBoxLayout(self)
        layout.addWidget(self.button)

    def start_thread(self):
        self.worker_thread.gui_text = self.button.text()
        self.worker_thread.start()

    def on_job_done(self, generated_str):
        print("Generated string : ", generated_str)
        self.button.setText(generated_str)


class WorkerThread(QtCore.QThread):

    job_done = QtCore.pyqtSignal('QString')

    def __init__(self, parent=None):
        super(WorkerThread, self).__init__(parent)
        self.gui_text = None

    def do_work(self):

        for i in range(0, 1000):
            print(self.gui_text)
            self.job_done.emit(self.gui_text + str(i))
            time.sleep(0.5)

    def run(self):
        self.do_work()


if __name__ == '__main__':
    app = QtWidgets.QApplication(sys.argv)
    test = MainWindow()
    test.show()
    app.exec_()

【讨论】:

谢谢,就像你说的代码 sn-p 工作正常。崩溃的原因是因为如果 try/except 块,我会从线程内部弹出警报。线程似乎不喜欢线程内的任何 UI 内容。当我改为打印到控制台时,一切正常,所以我需要弄清楚如何将该信息传递回 UI 类以显示给用户 @joke4me 您可以在 WorkerThread 中返回对象,job_done 可以将对象作为 QtCore.pyqtSignal(object) 返回。 @joke4me 你找到解决办法了吗?

以上是关于PyQt:如何从 QThread 获取 UI 数据的主要内容,如果未能解决你的问题,请参考以下文章

如何使用 PyQt5 QThread 从 CLASS(ShowVideo) 中获取值以在第二个 CLASS(ImageViewer) 中使用(根据我的程序)

PyQt5 OpenCV 网络摄像头使用 QThread

如何在Python PyQt中中断QThread上的脚本执行?

在 PyQt5 中取消 QThread

PyQt:如何终止可重用的 QThread

Python PyQt5 在 QThread 上为 QPushButton 加载图像