当前位置: 首页 > news >正文

PyQt——信号与槽

一.信号与槽的概念

        在PyQt中,信号(Signal)和槽(slot)是实现对象之间通信的核心机制,基于Qt的信号与槽机制。信号和槽是Qt框架中用于实现事件驱动编程的重要工具,允许对象之间以一种松耦合的方式进行交互。

1.信号(signal)

        信号就是当事件(按钮点击,内容改变,窗口关闭事件)或者是状态(check选中了,togglebutton切换)。当程序触发了某种状态或者发生了某种事件(如:按钮被点击了,内容改变等等),那么就可以发射出来一个信号。

2.槽(slot)

        槽是放置执行逻辑代码的一个函数,当信号发射过来后,捕获到信号就会执行与之绑定的槽内的函数。

3.信号与槽之间的连接

        为了实现信号和槽之间的功能,比如:当点击某个按钮需要执行某个逻时,需要把具体的信号和具体的槽函数绑定到一起。

sender.signal.connect(receiver.slot)
  • sender:发出信号的对象。

  • signal:信号名称,通常是内置信号(如 clicked())或自定义信号。

  • receiver:接收信号的对象。

  • slot:槽函数,用于处理信号。

二.信号和槽的示例

示例1:按钮接收信号

在 PyQt 中,按钮(如 QPushButton)本身并不“接收”信号,而是发出信号。按钮的信号(如 clicked)会在特定事件发生时被触发,并通知其他对象(通过槽函数)来处理这些事件。

注意:当按钮被点击时,clicked 信号会被触发,并传递一个布尔值参数。

import sys

from PyQt5.QtWidgets import QApplication, QWidget, QPushButton


class MyWindow(QWidget):
    def __init__(self):
        super().__init__()
        self.init_ui()

    def init_ui(self):
        # 更改当前Widge的宽高
        self.resize(500, 300)
        # 创建一个按钮
        btn = QPushButton("点我点我", self)
        # 设置窗口位置、宽高
        btn.setGeometry(200, 200, 100, 30)
        # 将按钮被点击时触发的信号与我们定义的函数(方法)进行绑定
        # 注意:这里没有(),即写函数的名字,而不是名字()
        btn.clicked.connect(self.click_my_btn)

    def click_my_btn(self, arg):
        # 槽函数,点击按钮则调用该函数
        # 这里的参数正好是信号发出,传递的参数
        print("点击按钮啦~", arg)


if __name__ == '__main__':
    app = QApplication(sys.argv)

    w = MyWindow()
    w.show()

    app.exec()

输出结果:

 

示例2:自定义信号

        在PyQt中,自定义信号是指开发者根据自己的需求定义的信号,用于PyQt应用程序中实现对象之间的通讯。自定义信号可以携带数据(如字符串,整数,列表等),并且可以在任何继承自QObject的类中定义和使用。


如何自定义信号?

自定义信号需要使用 pyqtSignal 装饰器定义,并且只能在继承自 QObject类中定义

 QObject是什么东西?

        QObject 是 Qt 框架的核心基类,提供了信号与槽机制、事件处理、对象管理和多线程支持等重要功能。在 PyQt 中,任何需要使用信号与槽机制的类都需要继承自 QObject。通过继承 QObjectWorker 类可以定义和发射自定义信号,从而实现与其他对象的通信。


import sys
from PyQt5.QtWidgets import QApplication, QWidget, QPushButton, QLabel, QVBoxLayout
from PyQt5.QtCore import QObject, pyqtSignal


class Worker(QObject):
    # 定义一个自定义信号,携带一个字符串参数
    custom_signal = pyqtSignal(str)

    def __init__(self, name):
        super().__init__()
        self.name = name

    def start_work(self):
        # 发射自定义信号
        self.custom_signal.emit(f"Hello from {self.name}!")


class MyWindow(QWidget):
    def __init__(self):
        super().__init__()
        self.init_ui()

    def init_ui(self):
        # 创建一个按钮
        button = QPushButton("Start Work", self)
        label = QLabel("Waiting...", self)

        # 创建 Worker 对象
        self.worker = Worker("Worker1")

        # 连接自定义信号到槽函数
        self.worker.custom_signal.connect(self.on_custom_signal)

        # 连接按钮点击信号到 Worker 的槽
        button.clicked.connect(self.worker.start_work)

        # 布局设置
        layout = QVBoxLayout()
        layout.addWidget(button)
        layout.addWidget(label)
        self.setLayout(layout)

        self.setWindowTitle("Custom Signal Example")
        self.label = label

    def on_custom_signal(self, message):
        # 槽函数,处理自定义信号
        self.label.setText(message)


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

输出结果:

理解:直接接收信号和自定义信号的区别

        我们看完上面的两个案例以后,会发现两个都是点击按钮啊,然后去触发信号,有样又有何区别呢?经过对比我们发现第一个案例是按钮触发,它是只传递布偶值的,而第二个案例我们可以同过自定义来规定我们要发送和接收的类型。


两者的区别

  1. 内置信号 vs 自定义信号

    • 内置信号:如 QPushButtonclicked 信号,是 PyQt 提供的现成信号,用于处理按钮点击事件。它只能传递固定类型的参数(如 clicked 信号传递一个布尔值)。

    • 自定义信号:通过 pyqtSignal 定义,可以传递任意类型的参数(如字符串、列表、字典等)。它允许开发者定义自己的事件通知机制。

  2. 应用场景

    • 内置信号:适用于常见的事件处理(如按钮点击、文本框内容变化等)。它们是 PyQt 框架预先定义好的,使用方便。

    • 自定义信号:适用于更复杂的场景,如跨线程通信、自定义对象之间的通信等。它们提供了更大的灵活性。

  3. 代码结构

    • 内置信号:直接连接到控件(如按钮)的事件上,逻辑简单。

    • 自定义信号:需要定义一个 QObject 的子类(如 Worker),并在该类中定义和发射信号。这使得代码结构更清晰,逻辑更模块化。


为什么使用自定义信号?

在你的第二个代码中,Worker 类的 start_work 方法被设计为一个独立的任务,它通过发射自定义信号来通知其他对象(如 MyWindow)任务的执行结果。这种方式的优点包括:

  1. 解耦Worker 类和 MyWindow 类之间没有直接调用关系,它们通过信号与槽机制通信,降低了代码耦合度。

  2. 灵活性:自定义信号可以传递任意类型的数据,使得任务的执行结果可以更灵活地传递给其他对象。

  3. 多线程支持Worker 类可以运行在单独的线程中,通过信号与槽机制安全地与主线程通信。

相关文章:

  • 橙心同步助手更新,,支持博客园、头条和语雀
  • 数据结构--队列(C语言实现)
  • 【UCB CS 61B SP24】Lecture 17 - Data Structures 3: B-Trees 学习笔记
  • EMO模型详解及代码复现
  • (保姆级教程)Windows系统本地部署通义万相2.1视频生成模型
  • C++双指针法(尺取法)原理及模板代码与例题
  • 基于springboot的酒店客房管理系统----数据库课程设计
  • 【 <一> 炼丹初探:JavaWeb 的起源与基础】之 JavaWeb的诞生:从 CGI 到 Servlet 的技术演进
  • VulnHub-DarkHole_2靶机搭建保姆级教程2025
  • C++杂记——RAII (Resource Acquisition Is Initialization)
  • 【Java项目】基于SpringBoot的会员制医疗预约服务管理信息系统
  • 算法 BFS搜爆路径问题
  • 深搜专题6:迷宫问题
  • Python爬虫:一文掌握PyQuery模块
  • 【漫话机器学习系列】109.线性无关(Linearly Independent)
  • Rust~String、str、str、String、Box<str> 或 Box<str>
  • 从零开始构建高效Spring Boot应用:实战案例与最佳实践
  • 【Linux】I/O操作
  • k8s学习记录:环境搭建二(基于Kubeadmin)
  • C语言(3)—循环、数组、函数的详解
  • 微软宣布将裁员3%
  • 回望乡土:对媒介化社会的反思
  • 迪奥部分客户数据遭泄露,公司称正持续展开调查
  • 商务部召开外贸企业圆桌会:全力为外贸企业纾困解难,提供更多支持
  • 80后莆田市文旅局长马骏登台与杨宗纬合唱,“演唱会秒变旅游推介会”
  • 人民币对美元即期汇率盘中创半年新高,离岸市场升破7.2