Сигнал PyQt5 между рабочим потоком и главным окном работает неправильно - PullRequest
0 голосов
/ 24 октября 2018

Я создаю приложение с PyQt5, у которого очень простой пользовательский интерфейс.Существует раскрывающийся список, из которого можно выбрать значение, а затем есть кнопка, по которой можно щелкнуть, и действие будет выполнено на основе текущего значения в раскрывающемся списке.

В настоящее время, если я запускаюкод ниже, я могу получить значение из выпадающего списка, и оно будет передано в рабочий поток до запуска myaction.myaction(customer_name).Код также работает нормально, но графический интерфейс не работает так, как я хочу, чтобы он работал, пока выполняется функция в рабочем потоке.Когда я нажимаю кнопку start , она должна подать сигнал в графический интерфейс пользователя, чтобы отключить кнопку, изменить ее метку и цвет, но этого никогда не происходит.Когда функция завершается, она должна быть возвращена к исходной форме.

Проблема в том, как я обрабатываю сигналы, или у меня избыточные функции в моих классах?Как правильно отправлять значение этого раскрывающегося списка в рабочий поток при каждом нажатии кнопки, чтобы я мог просто использовать его в качестве переменной там?

Каждое возможное решение, которое я нахожу в Интернете, вызывает у меня восторг, но ни одноиз них уже работали для меня, или некоторые из них были слишком запутанными, чтобы я мог понять.

Вот некоторые ответы, которые я уже прошел


#!/usr/bin/env python3

import sys
#import myaction
import time
from PyQt5.QtWidgets import QWidget, QApplication, QPushButton, QComboBox, QLabel
from PyQt5 import QtCore

class ConfWorker(QtCore.QThread):
    updated_button = QtCore.pyqtSignal(list)
    updated_label = QtCore.pyqtSignal(str)
    updated_error = QtCore.pyqtSignal(str)
    request_signal = QtCore.pyqtSignal()
    customer = QtCore.pyqtSignal(str)

    def __init__(self, parent=None):
        super(ConfWorker, self).__init__(parent)
        self.customer.connect(self.getcustomer)

    def run(self):
        self.request_signal.emit()

    def getcustomer(self, text):
        self.configure(text)

    def configure(self, customer_name):
        self.updated_button.emit(["In progress...", False])
        self.updated_label.emit(customer_name)
        time.sleep(5) # During this time you should be able to see color change etc.
        #myaction.myaction(customer_name)# TAKES ~20 SECONDS TO FINISH
        self.updated_button.emit(["Start", True])

class ConfGUI(QWidget):
    def __init__(self, parent=None):
        super(ConfGUI, self).__init__()

        self.worker = ConfWorker(self)
        self.worker.updated_button.connect(self.updateButton)
        self.worker.updated_label.connect(self.updateLabel)
        self.worker.updated_error.connect(self.updateError)
        self.worker.request_signal.connect(self.sendCustomer)

        self.targetBtn = QPushButton('Start Configuration', self)
        self.targetBtn.setStyleSheet("QPushButton { background-color: green; color: white }"
                        "QPushButton:disabled { background-color: red; color: white }")
        self.targetBtn.clicked.connect(self.worker.start)
        self.targetBtn.setGeometry(100, 400, 200, 50)
        self.setGeometry(800, 300, 400, 550)
        self.setFixedSize(400, 550)

        self.customerlist = QComboBox(self)
        self.customerlist.setGeometry(100, 50, 200, 50)
        self.customerlist.setObjectName("customer")
        self.customerlist.addItem("testcustomer1")
        self.customerlist.addItem("testcustomer2")
        self.customerlist.addItem("testcustomer3")

        self.label = QLabel(self)
        self.label.setText("")
        self.label.setStyleSheet('font-size: 30pt; font-family: Courier; color: green;')
        self.label.setGeometry(70,250,400,50)

        self.error_label = QLabel(self)
        self.error_label.setText("")
        self.error_label.setStyleSheet('font-size: 30pt; font-family: Courier; color: red;')
        self.error_label.setGeometry(70,350,400,50)

        self.show()

    def sendCustomer(self):
        self.worker.customer.emit(self.customerlist.currentText())

    def updateButton(self, button_list):
        self.targetBtn.setText(button_list[0])
        self.targetBtn.setEnabled(button_list[1])

    def updateLabel(self, label_text):
        self.label.setText(label_text)

    def updateError(self, error_text):
        self.error_label.setText(error_text)


if __name__ == '__main__':
    app = QApplication(sys.argv)
    ex = ConfGUI()
    sys.exit(app.exec_())

Ответы [ 2 ]

0 голосов
/ 26 октября 2018

Проблема вызвана очень распространенной и неправильной концепцией, они думают, что QThread является Qt Thread , то есть новым потоком, созданным Qt, но нет, QThread является потоком обработчик , как указано в документах :

Класс QThread обеспечивает независимый от платформы способ управления потоками.

Объект QThread управляетодин поток управления в программе.QThreads начинает выполняться в run ().По умолчанию run () запускает цикл обработки событий, вызывая exec () и запускает цикл обработки событий Qt внутри потока.

Единственная часть, которая выполняется в другом потоке, - это метод run, а весли вы не звоните, потому что ваша логика отличается, вы не хотите непрерывно выполнять тяжелую задачу, но по запросу пользователя, поэтому дизайн должен быть таким же, как у работника, но вы используете QThread в качестве базового класса, и это неверноВы должны использовать QObject в качестве базового класса и переместить его в новый поток, чтобы QObject выполнял свои задачи в этом потоке, предотвращая блокировку графического интерфейса.

QObject не является потоко-безопасным, он живетв потоке, и в потоке, где это определяется следующим образом:

  • A QObject живет в потоке родителя

  • Если выне иметь родителя, жить в потоке, в котором он был создан, если вы не перешли в другой поток.

  • И вы можете перейти в другой поток с помощью функции moveToThread(), но moveToThread() потерпит неудачу, если у QObject есть родитель, поскольку первый критерий является привилегированным.

С другой стороны, если вы хотите вызвать метод из другого потока, необходимо использоватьДекоратор @QtCore.pyqtSlot()


Учитывая вышеизложенное, мы получаем следующее решение:

#!/usr/bin/env python3
import sys
import time
from PyQt5 import QtCore, QtWidgets


class ConfWorker(QtCore.QObject):
    updated_button = QtCore.pyqtSignal(list)
    updated_label = QtCore.pyqtSignal(str)
    updated_error = QtCore.pyqtSignal(str)
    request_signal = QtCore.pyqtSignal()
    customer = QtCore.pyqtSignal(str)

    def __init__(self, parent=None):
        super(ConfWorker, self).__init__(parent)
        self.customer.connect(self.getcustomer)

    @QtCore.pyqtSlot()
    def doWork(self):
        self.request_signal.emit()

    @QtCore.pyqtSlot(str)
    def getcustomer(self, text):
        self.configure(text)

    def configure(self, customer_name):
        self.updated_button.emit(["In progress...", False])
        self.updated_label.emit(customer_name)
        time.sleep(5) # During this time you should be able to see color change etc.
        #myaction.myaction(customer_name)# TAKES ~20 SECONDS TO FINISH
        self.updated_button.emit(["Start", True])

class ConfGUI(QtWidgets.QWidget):
    def __init__(self, parent=None):
        super(ConfGUI, self).__init__()

        # create a QThread and start the thread that handles
        thread = QtCore.QThread(self)
        thread.start()

        # create the worker without a parent so you can move
        self.worker = ConfWorker()
        # the worker moves to another thread
        self.worker.moveToThread(thread)

        self.worker.updated_button.connect(self.updateButton)
        self.worker.updated_label.connect(self.updateLabel)
        self.worker.updated_error.connect(self.updateError)
        self.worker.request_signal.connect(self.sendCustomer)

        self.targetBtn = QtWidgets.QPushButton('Start Configuration', self)
        self.targetBtn.setStyleSheet("QPushButton { background-color: green; color: white }"
                        "QPushButton:disabled { background-color: red; color: white }")
        self.targetBtn.clicked.connect(self.worker.doWork)
        self.targetBtn.setFixedSize(200, 50)

        self.customerlist = QtWidgets.QComboBox(self)
        self.customerlist.addItems(["testcustomer1", "testcustomer2", "testcustomer3"])
        self.customerlist.setFixedSize(200, 50)

        self.label = QtWidgets.QLabel(self, alignment=QtCore.Qt.AlignCenter)
        self.label.setStyleSheet('font-size: 30pt; font-family: Courier; color: green;')
        self.label.setFixedSize(400,50)

        self.error_label = QtWidgets.QLabel(self, alignment=QtCore.Qt.AlignCenter)
        self.error_label.setStyleSheet('font-size: 30pt; font-family: Courier; color: red;')
        self.error_label.setFixedSize(400,50)

        lay = QtWidgets.QVBoxLayout(self)
        lay.addWidget(self.customerlist, alignment=QtCore.Qt.AlignCenter)
        lay.addWidget(self.label, alignment=QtCore.Qt.AlignCenter)
        lay.addWidget(self.error_label, alignment=QtCore.Qt.AlignCenter)
        lay.addWidget(self.targetBtn, alignment=QtCore.Qt.AlignCenter)
        self.setFixedSize(400, 550)

    @QtCore.pyqtSlot()
    def sendCustomer(self):
        self.worker.customer.emit(self.customerlist.currentText())

    @QtCore.pyqtSlot(list)
    def updateButton(self, button_list):
        self.targetBtn.setText(button_list[0])
        self.targetBtn.setEnabled(button_list[1])

    @QtCore.pyqtSlot(str)
    def updateLabel(self, label_text):
        self.label.setText(label_text)

    @QtCore.pyqtSlot(str)
    def updateError(self, error_text):
        self.error_label.setText(error_text)


if __name__ == '__main__':
    app = QtWidgets.QApplication(sys.argv)
    ex = ConfGUI()
    ex.show()
    sys.exit(app.exec_())

Наблюдение: Также, как вы можете видеть в моем решении, QThread будет дочерним элементом окна, потому что QThread - это QObject, который обрабатывает другой поток.

0 голосов
/ 25 октября 2018

Вы действительно пишете очень сложные простые вещи.Попробуйте:

import sys
from PyQt5.QtWidgets import QWidget, QApplication, QPushButton, QComboBox, QLabel
from PyQt5.QtCore    import QThread, pyqtSignal

class ConfWorker(QThread):
    threadSignal = pyqtSignal(str)
    finishSignal = pyqtSignal(str)

    def __init__(self, startParm):
        super().__init__()
        self.startParm = startParm   # Initialize the parameters passed to the task 

    def run(self):
        # Do something...
        for i in range(20):
            text = "In progress ................." \
                    if i%2==0 else "In progress {}".format(self.startParm)
            self.threadSignal.emit(text)
            QThread.msleep(500)
        self.finishSignal.emit(self.startParm)


class ConfGUI(QWidget):
    def __init__(self):
        super().__init__()
        self.setGeometry(800, 100, 400, 550)
        self.setFixedSize(400, 550)        

        self.targetBtn = QPushButton('Start Configuration', self)
        self.targetBtn.setStyleSheet(
                        "QPushButton { background-color: green; color: white;}"
                        "QPushButton:disabled { background-color: red; color: white;}"
                        )
        self.targetBtn.setGeometry(100, 400, 200, 50)
        self.targetBtn.clicked.connect(self.workerStart)           

        self.customerlist = QComboBox(self)
        self.customerlist.setGeometry(100, 50, 200, 50)
        self.customerlist.setObjectName("customer")
        self.customerlist.addItem("testcustomer1")
        self.customerlist.addItem("testcustomer2")
        self.customerlist.addItem("testcustomer3")
        self.customerlist.activated[str].connect(self.comboActivated)
        self.startParm = "testcustomer1"

        self.label = QLabel(self)
        self.label.setStyleSheet('font-size: 30pt; font-family: Courier; color: green;')
        self.label.setGeometry(70,250,400,50)

    def workerStart(self):
        self.targetBtn.setText("In progress...")
        self.targetBtn.setEnabled(False)
        self.label.setText("")
        self.worker = ConfWorker(self.startParm)  
        self.worker.threadSignal.connect(self.on_threadSignal)
        self.worker.finishSignal.connect(self.on_finishSignal)        
        self.worker.start()                     

    def on_threadSignal(self, text):
        self.targetBtn.setText(text)

    def on_finishSignal(self, text):
        self.targetBtn.setEnabled(True)
        self.targetBtn.setText("Start Configuration'")
        self.label.setText(text)

    def comboActivated(self, text):
        self.startParm = text


if __name__ == '__main__':
    app = QApplication(sys.argv)
    ex  = ConfGUI()
    ex.show()
    sys.exit(app.exec_())

enter image description here

...