前言
在 Qt 开发中,信号与槽(Signal & Slot) 是最核心、最独特的通信机制。它让原本独立的控件能够"对话",让程序的事件响应变得清晰而优雅。
1. 为什么需要信号与槽?
想象一个场景:界面上有一个按钮和一个窗口。我们希望点击按钮后窗口关闭。
-
按钮不知道窗口的存在,窗口也不知道按钮何时被点击。
-
传统的回调函数方式耦合性强,难以维护。
Qt 给出的答案是:按钮发出"我被点击了"的信号,窗口用自己的"关闭"槽函数来响应这个信号 。两者通过 connect 连接,彼此独立,却又协同工作。
2. 信号与槽是什么?
2.1 信号(Signal)
信号的本质就是"事件"。用户对控件进行操作(点击、移动、键盘输入等),控件内部就会产生对应的信号。
例如:
-
按钮被单击 → 发出
clicked()信号 -
窗口被关闭 → 发出
destroyed()信号 -
鼠标移动 → 发出
mouseMove()信号
信号的呈现形式是成员函数 ,但不需要我们实现 (由 Qt 的元对象编译器 moc 自动生成)。我们只需要在类的 signals: 区域声明即可。


2.2 槽(Slot)
槽的本质就是"响应函数"。当某个信号被触发时,与之关联的槽函数会自动执行。
-
槽就是一个普通的 C++ 函数,可以放在 public slots:、protected slots: 或 private slots: 中(高版本 Qt 也允许放在普通 public 下)。
-
槽函数需要我们自己实现(定义函数体)。
-
槽可以带参数,也可以重载,但不能有默认参数。
简单记忆:信号是"发生了什么",槽是"对此做什么"。
3. 信号与槽的工作流程(核心原理)
下面的流程图直观展示了信号与槽的协作过程:

实际底层是通过函数间的相互调用 实现的。例如 clicked() 信号函数内部会调用 close() 槽函数,但这一切对开发者是透明的。

4. 如何使用信号与槽?

4.1 手动连接:connect() 函数
Qt 提供了 QObject::connect() 静态函数,用于将信号和槽关联起来。其原型如下(Qt5 常用语法):
connect(sender, &Sender::signal, receiver, &Receiver::slot);

参数说明:
-
sender:信号的发送者(对象指针) -
signal:信号函数的地址(如&QPushButton::clicked) -
receiver:信号的接收者(对象指针) -
slot:槽函数的地址(如&QWidget::close)
示例:点击按钮关闭窗口
QPushButton *btn = new QPushButton("关闭", this);
connect(btn, &QPushButton::clicked, this, &QWidget::close);
注意:信号和槽的参数个数、类型必须匹配(或信号参数可多于槽,但实际开发建议保持一致)。


代码示例:在窗口中设置一个按钮,当点击 "按钮" 时 , 关闭"窗口"。

4.2 如何找到内置的信号和槽?
Qt 提供了丰富的内置控件,它们自带的信号和槽可以通过Qt 帮助文档查阅。


以 QPushButton 为例:
-
在帮助文档中搜索
QPushButton。 -
若本类没有
signals关键字,则去其父类(如QAbstractButton)中查找。 -
父类中列出了常见信号:
clicked(bool),pressed(),released(),toggled(bool)。

槽函数的查找方式相同,关键字为 slots。
小技巧:
clicked(bool)中的bool参数对于普通按钮无意义,通常用clicked()即可;复选框等"可切换"控件才会使用带参数的版本。
4.3 通过QT Creator生成槽代码
QT Creator可以快速帮助我们生成信号槽相关的代码。
对于初学者,Qt Creator 提供了便捷的可视化操作,自动生成槽函数框架,省去手动 connect 的烦恼。
步骤详解
① 新建项目(记得勾选"生成 UI 设计文件")
② 打开 widget.ui 文件,进入设计界面
③ 拖入一个按钮 ,并修改其显示文字和 objectName(例如 pushButton)
④ 右键按钮 → 选择"转到槽..."
⑤ 在弹出的对话框中选择 clicked() 信号(普通按钮选择此即可)
⑥ 自动生成代码:
-
在
widget.h中会自动添加槽函数声明:private slots:
void on_pushButton_clicked();






自动命名规则 :on_XXX_SSS
-
on_:固定前缀 -
XXX:控件的objectName -
SSS:信号名
例如:on_pushButton_clicked() 表示 pushButton 控件的 clicked 信号
⚠️ 重要建议 :日常编码中,优先使用显式
connect,而非依赖自动命名规则。这样可以:
代码意图更清晰
避免拼写错误导致连接失效

