当前位置: 首页 > 面试题库 >

PyQt:将信号连接到插槽以启动后台操作

谭伟
2023-03-14
问题内容

我有以下代码,该代码scan_value在更新ui(progress)中的进度条时执行后台操作()。scan_value迭代in中的某个值objvalue_changed每次更改该值都会发出一个信号()。由于此处不相关的原因,我必须将其包装Scanner在另一个线程的对象()中。当一个按钮扫描仪被称为scanclicked。这是我的问题……以下代码可以正常工作(即进度条会及时更新)。

# I am copying only the relevant code here.

def update_progress_bar(new, old):
    fraction = (new - start) / (stop - start)
    progress.setValue(fraction * 100)

obj.value_changed.connect(update_progress_bar)

class Scanner(QObject):

    def scan(self):
        scan_value(start, stop, step)
        progress.setValue(100)

thread = QThread()
scanner = Scanner()
scanner.moveToThread(thread)
thread.start()

scan.clicked.connect(scanner.scan)

但是,如果我将最后一部分更改为:

thread = QThread()
scanner = Scanner()
scan.clicked.connect(scanner.scan) # This was at the end!
scanner.moveToThread(thread)
thread.start()

进度条仅在最后更新(我猜是所有东西都在同一线程上运行)。在将对象接收对象移动到线程之前或之后将信号连接到插槽是否无关紧要。


问题答案:

连接是在将工作对象移动到另一个线程之前还是之后都没有关系。引用Qt docs:

Qt :: AutoConnection- 如果信号是从与接收对象不同的线程发出的,则将信号排队,表现为 Qt ::
QueuedConnection
。否则,将以 Qt :: DirectConnection的形式 直接调用该插槽。
连接的类型在发出信号时确定 。[重点添加]

因此,只要将的type参数connect设置为QtCore.Qt.AutoConnection(这是默认值),Qt就应确保以适当的方式发出信号。

示例代码的问题很可能是 时隙而 不是 信号
。信号连接到的python方法可能需要使用pyqtSlot装饰器标记为Qt插槽:

from QtCore import pyqtSlot

class Scanner(QObject):

    @pyqtSlot()
    def scan(self):
        scan_value(start, stop, step)
        progress.setValue(100)

编辑

应该澄清的是,只有在最近的Qt版本中,连接类型才在发出信号时确定。在版本4.4中引入了此行为(以及Qt多线程支持中的其他一些更改)。

同样,在PyQt特定问题上可能值得进一步扩展。在PyQt中,信号可以连接到Qt插槽,另一个信号或任何可调用的python(包括lambda函数)。对于后一种情况,将在内部创建一个代理对象,该对象包装可调用的python,并提供Qt信号/插槽机制所需的插槽。

导致此问题的正是此代理对象。创建代理后,PyQt只需执行以下操作:

    if (rx_qobj)
        proxy->moveToThread(rx_qobj->thread());

如果 将接收对象移动到其线程 之后 进行连接,则很好;但如果 之前完成 ,则代理将保留在主线程中。

使用@pyqtSlot装饰器可以完全避免此问题,因为它可以更直接地创建Qt插槽,并且根本不使用代理对象。

最后,还应注意,此问题当前不影响PySide。



 类似资料:
  • 问题内容: 我有一个任务栏菜单,单击该任务栏菜单将其连接到获取触发事件的插槽。现在的问题是,我想知道单击了哪个菜单项,但是我不知道如何将该信息发送到连接的功能。这是用于将操作连接到功能的代码: 我知道有些事件会返回一个值,但是Trigger()不会。那么我该如何实现呢?我必须发出自己的信号吗? 问题答案: 用一个 这是PyQt书中的一个示例: 顺便说一句,您也可以使用,但是我发现该方法更加简单明了

  • 信号与槽可以通过使用手写代码显式的实现关联 ,也可以运用 QMetaObject 类规定的槽 函数命名范式来实现自动关联。 10.5.1 显式关联 首先我们来看一下,不使用“自动关联规则”的情形。 在下面这段代码里面,我们定义了一个对话框类,它有一个私有的槽 checkValues(), 它用来检验用户提供的值是否正确。 class ImageDialog : public QDialog, pr

  • 问题内容: 某些小部件将允许我执行以下操作: 但这样做: 无法说对象没有属性“连接”。 我知道子类化小部件并重新实现该方法将使我能够响应事件。但是,如何从用户上下文来看此后的键盘事件或以其他方式说呢? 问题答案: 创建一个自定义信号,并从重新实现的事件处理程序中发出它: (注意:为了保持事件的现有处理,需要调用基类实现)。

  • 与以顺序方式执行的控制台模式应用程序不同,基于GUI的应用程序是事件驱动的。 执行函数或方法以响应用户的操作,例如单击按钮,从集合中选择项目或鼠标单击等,称为events 。 用于构建GUI界面的窗口小部件充当此类事件的来源。 每个PyQt小部件都是从QObject类派生的,旨在发出“ signal ”以响应一个或多个事件。 信号本身不会执行任何操作。 相反,它“连接”到“ slot ”。 插槽可

  • 我的测试类: 所以,我这样使用我的类: 我在signal.cpp中得到一个警告:候选函数不可行:第一个参数没有从'void(*)(int)'到'int'的已知转换; 在此代码中: 我该怎么解决?如果我想给我的“emit”方法一个对象或多个参数,我可以做什么? 谢了!

  • 信号和槽机制是 Qt 的核心机制之一,要掌握 Qt 编程就需要对信号和槽有所了解。信号和槽是一种高级接口,它们被应用于对象之间的通信,它们是 Qt 的核心特性,也是 Qt不同于其它同类工具包的重要地方之一。 在我们所了解的其它 GUI 工具包中,窗口小部件(widget)都有一个回调函数用于响应 它们触发的动作,这个回调函数通常是一个指向某个函数的指针。在 Qt 中用信号和槽取代 了上述机制。 1