python - 向 PyQt5 中的 Thread 函数发送信号导致装饰槽的 TypeError

标签 python multithreading pyqt5 signals-slots qthread

我目前正在开发一个 Gui,它会一直计数直到达到给定的输入数字。应该可以在计数期间停止 while 循环,并使用按钮再次重新启动它,而无需结束主 Gui 线程。这已经可以工作了,但前提是目标计数在线程工作函数中是固定的,例如n = 10 -> 数到 10,并且在不关闭 Gui 并更改主代码中的数字的情况下无法更改为 20 或任何其他数字。我想使用行编辑进行数字输入,并将所需的目标数字发送到线程,以便在我开始计数时计数到给定的数字。但是,当启动 Gui(在实现发送和获取输入值通常需要的信号之后)在行编辑中键入数字并按开始按钮时,我收到一个 TypeError,指出“装饰槽没有与以下兼容的签名” started()'('started'应该连接线程和工作函数)。有谁知道我在哪里犯了错误或必须更改某些内容和/或是否有解决此错误的方法?

我非常绝望(我已经尝试解决这个问题很长一段时间了,但在网上找不到任何提示......),非常感谢任何帮助!致以诚挚的问候和感谢!

这是带有注释的完整代码:

import sys
import time
from PyQt5.QtWidgets import QPushButton, QMainWindow, QApplication, QLineEdit
from PyQt5.QtCore import QObject, QThread, pyqtSignal, pyqtSlot

class Worker(QObject):

    finished = pyqtSignal()  # signal out to main thread to alert it that work is completed

    def __init__(self):
        super(Worker, self).__init__()
        self.working = True  # flag to control our loop

    @pyqtSlot(int) # should take the sended integer n from signal...
def work(self, n):

        s = 0

        while self.working:

            if s != n: # count until input integer is reached (doesn't work so far)
                print(s)
                s += 1
                time.sleep(0.5)

        self.finished.emit() # alert gui that the loop stopped

class Window(QMainWindow):

    sendnumber = pyqtSignal(int) # send signal (this is how it is usually done...?)

    def __init__(self):

        super(Window, self).__init__()
        self.setGeometry(50, 50, 200, 250)
        self.setWindowTitle("Program")

        self.inputnumber=QLineEdit(self, placeholderText="number")
        self.inputnumber.resize(self.inputnumber.minimumSizeHint())
        self.inputnumber.move(50, 50)

        self.startbtn = QPushButton("Start", self)
        self.startbtn.resize(self.startbtn.minimumSizeHint())
        self.startbtn.move(50, 100)

        self.stopbtn = QPushButton("Stop", self)
        self.stopbtn.resize(self.stopbtn.minimumSizeHint())
        self.stopbtn.move(50, 150)

        self.thread = None
        self.worker = None

        self.startbtn.clicked.connect(self.start_loop)  

    def start_loop(self):

        self.thread = QThread()  # a new thread to run the background tasks in
        self.worker = Worker()  # a new worker to perform those tasks
        self.worker.moveToThread(self.thread)  # move the worker into the thread, do this first before connecting the signals

        self.thread.started.connect(self.worker.work)  # begin worker object loop when the thread starts running
        self.sendnumber.connect(self.worker.work) # connect input number to worker in thread
        # this doesn't work so far and gives a TypeError!

        self.stopbtn.clicked.connect(self.stop_loop)  # stop the loop on the stop button click
        self.worker.finished.connect(self.loop_finished)  # do something in the gui when the worker loop ends
        self.worker.finished.connect(self.thread.quit)  # tell the thread it's time to stop running
        self.worker.finished.connect(self.worker.deleteLater)  # have worker mark itself for deletion
        self.thread.finished.connect(self.thread.deleteLater)  # have thread mark itself for deletion
        # make sure those last two are connected to themselves or you will get random crashes

        self.thread.start()

    def stop_loop(self):

        self.worker.working = False
        # when ready to stop the loop, set the working flag to false

    @pyqtSlot() # as far as I know you need this Slot to send the input to Slot in thread?
    def getnumber(self):

        try:
            n = int(self.inputnumber.text()) # input for the work function in thread
            print('Trying number...')
        except:
            print('Use an integer!')
            return

        self.sendnumber.emit(n) # emit integer signal
        print('Emitting signal to worker...')

    def loop_finished(self):
        # received a callback from the thread that it's completed
        print('Loop finished.')

if __name__ == '__main__':
    def run():

        app = QApplication(sys.argv)
        gui = Window()
        gui.show()
        app.exec_()

    run() 

这是显示错误的控制台输出:

Traceback (most recent call last):

  File "C:/Users/***/WhileLoopInterruptTest2.py", line 63, in start_loop
    self.thread.started.connect(self.worker.work)

TypeError: decorated slot has no signature compatible with started()

这就是所有错误提示,并且必须关闭 Gui。 我使用的是 Python 3.6.1、Spyder 3.3.1、PyQt5.9。

最佳答案

替换这一行:

self.thread.started.connect(self.worker.work)

这样:

self.thread.started.connect(lambda: self.worker.work(10))

您可以将 10 替换为文本框中的值。只需记住首先将其转换为 int 即可。

结账 this article以获得更详细的解释。

关于python - 向 PyQt5 中的 Thread 函数发送信号导致装饰槽的 TypeError,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51894098/

相关文章:

linux - 如何在 linux 上监控进程的线程数?

python - 让 Enter 在 QAbstractTableModel tableView 中表现为 key_down

python - MySQL 'replace into select from' 替代方案

python - 如何将两列 Pandas 数据框移动和堆叠成一列?

java - Spring中@Scheduled方法中的Thread.sleep

python - PyCharm 3.0 找不到 PyQt5

python - 如何在 QLabel 中设置文本并显示 '<>' 个字符?

python - 生成器与序列对象

python - 从文本文件中读取数据并将其分配给数据框的最快方法是什么?

java - java ExecutorService newSingleThreadExecutor 是否仅使用一个线程执行所有任务?