C++ Qt开发:如何使用信号与槽

在Qt中,信号与槽(Signal and Slot)是一种用于对象之间通信的机制。是Qt框架引以为傲的一项机制,它带来了许多优势,使得Qt成为一个强大且灵活的开发框架之一。信号与槽的关联通过QObject::connect函数完成。这样的机制使得对象能够以一种灵活而松散耦合的方式进行通信,使得组件之间的交互更加灵活和可维护。

信号(Signal) 是一种特殊的成员函数,用于表示某个事件的发生。当特定的事件发生时,对象会发射(emit)相应的信号。例如,按钮被点击、定时器时间到达等都可以是信号。

槽(Slot) 是用于处理信号的成员函数。槽函数定义了在特定信号发生时执行的操作。一个槽可以与一个或多个信号关联,当信号被发射时,与之关联的槽函数将被调用。

在早期,对象间的通信采用回调实现。回调实际上是利用函数指针来实现,当我们希望某件事发生时处理函数能够获得通知,就需要将回调函数的指针传递给处理函数,这样处理函数就会在合适的时候调用回调函数。回调有两个明显的缺点:

  • 它们不是类型安全的,无法保证处理函数传递给回调函数的参数都是正确的。
  • 回调函数和处理函数紧密耦合,源于处理函数必须知道哪一个函数被回调。

而信号与槽机制则可以更好的比秒上述问题的产生,以下是信号与槽机制的一些优势:

  1. 松散耦合(Loose Coupling): 信号与槽机制实现了松散耦合,使得对象之间的连接更加灵活。对象不需要知道彼此的具体实现,只需通过信号与槽进行通信。这降低了组件之间的依赖关系,提高了代码的可维护性。
  2. 事件驱动(Event-Driven): 信号与槽机制使得Qt应用程序能够轻松地处理事件。例如,按钮的点击、定时器的超时等都可以通过信号与槽来处理,使得应用程序能够响应用户交互和外部事件。
  3. 模块化设计: 通过信号与槽,不同模块之间可以通过事件进行通信,这样可以更容易地设计和维护模块化的代码。一个模块的改变不太可能影响到其他模块,从而提高了代码的可维护性。
  4. 异步通信: 信号与槽机制支持跨线程的异步通信。当信号与槽连接在不同线程的对象上时,Qt会自动进行线程间的通信,使得开发者能够更方便地处理多线程应用。
  5. 灵活的连接方式: Qt支持多种连接方式,包括在代码中使用QObject::connect连接,也可以使用Qt Creator等工具在图形界面上进行可视化的信号与槽关联。这种灵活性使得开发者可以选择最适合他们需求的连接方式。
  6. 类型安全的连接(Qt5新增特性): 在Qt5中引入了新的connect语法,不再需要使用SIGNAL()和SLOT()宏,而是使用函数指针直接进行连接,从而在编译时进行类型检查,减少了潜在的运行时错误。

总体而言,这些优势使得Qt成为构建各种类型应用程序的理想选择。

1.1 信号与槽函数

1.1.1 Connect

信号和槽进行关联使用的是QObject类的connect()函数,QObject::connect 是用于建立信号与槽连接的Qt框架函数。它有几个不同的重载形式,但最常用的形式是:

static QMetaObject::Connection QObject::connect(const QObject *sender,const char *signal,const QObject *receiver,const char *slot,Qt::ConnectionType type = Qt::AutoConnection
);

参数解释如下:

  • sender:发出信号的对象指针。
  • signal:信号的签名,使用 SIGNAL 宏包装,指定了发出的信号。
  • receiver:接收信号的对象指针。
  • slot:槽函数的签名,使用 SLOT 宏包装,指定了接收到信号时要调用的函数。
  • type:连接的类型,是一个枚举值,可以是 Qt::AutoConnectionQt::DirectConnectionQt::QueuedConnectionQt::BlockingQueuedConnection

在函数定义中,第一个参数sender为发送信号的对象,第二个参数signal为要发送的信号,第三个参数receiver为接收信号的对象,第4个参数slot为接收对象在接收到信号之后所需要调用的槽函数。该函数的最后一个参数表明了关联的方式,默认值是Qt::AutoConnection方式,函数最终返回值是一个 QMetaObject::Connection 对象,可以用于断开连接时使用。

这个函数的作用是将 sender 对象的 signalreceiver 对象的 slot 进行连接。当 sender 发出信号时,receiver 对象的 slot 函数将被调用。

1.1.2 Disconnect

QObject::disconnect 是 Qt 框架用于断开信号与槽连接的函数。它有几个不同的重载形式,但最常用的形式是:

static bool QObject::disconnect(const QObject *sender,const char *signal,const QObject *receiver,const char *slot
);

参数解释如下:

  • sender:发出信号的对象指针。
  • signal:信号的签名,使用 SIGNAL 宏包装,指定了发出的信号。
  • receiver:接收信号的对象指针。
  • slot:槽函数的签名,使用 SLOT 宏包装,指定了接收到信号时要调用的函数。

这个函数的作用是断开 sender 对象的 signalreceiver 对象的 slot 之间的连接。如果连接存在,那么它将被断开,不再触发。该函数返回值是一个 bool 类型,表示是否成功断开连接。

1.2 应用信号与槽

1.2.1 信号与槽绑定

信号与槽函数的使用非常容易理解,笔者将以最简单的案例来告诉大家该如何灵活的运用这两者,首先新建一个Qt Widgets Application项目,如下图所示第一个则是该项目的选项卡,其他参数保持默认即可;

当项目被创建好之后读者应该能构建看到如下图所示的页面提示信息,其中的untitled.pro是项目的主配置文件该配置文件一般有Qt自动维护,文件夹Headers则是项目的头文件包含路径,Sources则是代码的实现路径,最后一个Forms是用于图形化设计的UI模板。

首先双击mainwindow.ui进入到UI设计模式,接着拖拽一个PushButton按钮组件,与两个lineEdit组件到右侧的窗体画布上,并按下Ctrl+S保存该画布,刷新配置文件,如下图所示;

此时回到编辑菜单,并点击mainwindow.h头文件部分,并在头文件mainwindow.h的类MainWindow的定义中声明槽函数,代码如下,其含义是定义一个按钮点击槽:

public slots:void on_pushButton_clicked();

接着我们就需要点击mainwindow.cpp文件,并在头文件中实现这个槽函数的具体功能,此处我们就实现设置两个lineEdit组件分别用于显示两串字符串,代码如下;

void MainWindow::on_pushButton_clicked()
{ui->lineEdit->setText("hello lyshark");ui->lineEdit_2->setText("www.lyshark.com");
}

最后一步则是建立映射关系,在类MainWindow的构造函数中添加如下语句,以便将信号和槽函数进行连接:

#include "mainwindow.h"
#include "ui_mainwindow.h"MainWindow::MainWindow(QWidget *parent): QMainWindow(parent), ui(new Ui::MainWindow)
{ui->setupUi(this);// 建立关联当点击pushButton时信号clicked 发送给槽on_pushButton_clickedconnect(ui->pushButton,SIGNAL(clicked()),this,SLOT(on_pushButton_clicked));
}MainWindow::~MainWindow()
{delete ui;
}void MainWindow::on_pushButton_clicked()
{ui->lineEdit->setText("hello lyshark");ui->lineEdit_2->setText("www.lyshark.com");
}

此时运行程序,当读者点击按钮时,则会自动触发on_pushButton_clicked()所关联的代码,将两个lineEdit设置为不同的内容,如下图;

当然了,上述过程都是需要我们手动的去关联信号与槽,在开发中其实可以直接在PushButton组件上邮件,选中转到槽选项,此时则会弹出关于该组件所支持的所有槽函数,读者只需要选中并双击,即可自动实现槽函数的创建与管理,这对于高效率开发是至关重要的。

当然在槽函数使用结束后我们需要断开,在断开时直接使用disconnect并传入需要断开的绑定sender信号即可,如下所示;

void MainWindow::on_pushButton_2_clicked()
{disconnect(ui->pushButton,SIGNAL(clicked()),nullptr,nullptr);
}

1.2.2 匿名函数绑定

你是否感觉使用代码创建信号与槽很麻烦呢,其实通过使用Lambda表达式我们可以与Connect完美的结合在一起使用,者能够让信号与槽的使用更加的得心应手。

Lambda表达式是一种匿名函数的表示方式,引入C++11标准,用于创建内联函数或闭包。Lambda表达式可以在需要函数对象的地方提供一种更为简洁和灵活的语法。

它的基本形式如下:

[capture](parameters) -> return_type {// 函数体
}
  • capture:用于捕获外部变量的列表。可以捕获外部变量的值或引用,也可以省略不捕获任何变量。捕获列表是Lambda表达式的一部分。
  • parameters:参数列表,类似于普通函数的参数。
  • return_type:返回类型,指定Lambda表达式的返回类型。可以省略,由编译器自动推断。
  • {}:Lambda表达式的函数体。

使用Lambda表达式与Qt的connect函数结合实现匿名槽函数。具体概述如下:

Lambda表达式的初始化

[=]() {this->setWindowTitle("初始化..");
}();

这里使用Lambda表达式对 this->setWindowTitle("初始化.."); 进行了初始化,Lambda表达式中的 [=] 表示捕获外部变量并通过值传递,其中的 () 表示Lambda表达式立即执行,实现对窗口标题的初始化。

Lambda表达式作为槽函数

connect(btn_ptr1, &QPushButton::clicked, this, [=]() mutable {number = number + 100;std::cout << "inner: " << number << std::endl;
});

这里使用Lambda表达式作为 btn_ptr1 按钮的槽函数。在Lambda表达式中,使用了 mutable 关键字,允许修改通过值传递的变量 number。当按钮 btn_ptr1 被点击时,Lambda表达式内部修改了 number 的值,并输出修改后的值。

Lambda表达式中的返回值

int ref = []() -> int {return 1000;
}();
std::cout << "Return = " << ref << std::endl;

这里的Lambda表达式中带有返回值的情况。Lambda表达式通过 -> int 指定返回类型,然后在大括号中返回了一个整数值。该Lambda表达式被立即执行,返回值被赋给变量 ref,并输出到控制台。

如下,我们就来演示一个简单的直接使用匿名函数实现功能的案例,当使用匿名函数时,只需要在Connect时将功能一并写到链接函数的底部即可,此时的效果等同于上述功能,因为没有函数名所以显得更加简单,如下图;

MainWindow::MainWindow(QWidget *parent): QMainWindow(parent), ui(new Ui::MainWindow)
{ui->setupUi(this);// 匿名函数connect(ui->pushButton,&QPushButton::clicked,this,[=](){std::cout << "hello lyshark" << std::endl;ui->lineEdit->setText("www.lyshark.com");});
}

总体来说,匿名函数(Lambda表达式)在Qt中与connect函数一起使用,提供了一种方便的方式来定义简短的槽函数,使得代码更加紧凑和可读。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/264967.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

如何使用 Explain 分析 SQL 语句?

如何使用 Explain 分析 SQL 语句&#xff1f; MySQL中EXPLAIN命令是我们分析和优化SQL语句的利器。 如何使用EXPLAIN来分析SQL语句&#xff0c;接下来有15个例子&#xff0c;一起学习呗 1. EXPLAIN的基本使用 EXPLAIN可以用于分析MySQL如何执行一个SQL查询&#xff0c;包括如…

css的Grid布局

1.简单布局 .grid { display: grid; grid-template-columns: 1fr 2fr 1fr; 布局样式 column-gap: 24px; 列间距 row-gap: 24px; 行间距 } 2.排列布局 center垂直方向居中对其 end靠下对齐 3.水平方向对齐 center居中 end靠右对齐 space-between两段对齐 4.对…

你知道LOL中点地面移动是怎么实现的吗?

引言 Cocos中点地面移动的实例。 在游戏开发中&#xff0c;我们经常会遇到通过点击地面控制玩家移动到指定点的需求。 本文将介绍一下如何在Cocos中实现类似LOL的点地面移动效果。 本文源工程在文末获取&#xff0c;小伙伴们自行前往。 点地面移动知识点 要在Cocos中实现类…

【Hadoop_03】HDFS概述与Shell操作

1、集群配置&#xff08;1&#xff09;集群启动/停止方式总结&#xff08;2&#xff09;编写Hadoop集群常用脚本&#xff08;3&#xff09;常考面试题【1】常用端口号【2】常用配置-文件 2、HDFS概述&#xff08;1&#xff09;HDFS产出背景及定义&#xff08;2&#xff09;HDFS…

数字化转型怎么才能做成功?_光点科技

数字化转型对于现代企业来说是一场必要的革命。它不仅仅是技术的更迭&#xff0c;更是企业战略、文化和运营方式全面升级的体现。一个成功的数字化转型能够使企业更具竞争力、更灵活应对市场变化&#xff0c;并最终实现业务增长和效率提升。那么&#xff0c;数字化转型怎么才能…

【开源】基于Vue+SpringBoot的免税店商城管理系统

文末获取源码&#xff0c;项目编号&#xff1a; S 069 。 \color{red}{文末获取源码&#xff0c;项目编号&#xff1a;S069。} 文末获取源码&#xff0c;项目编号&#xff1a;S069。 目录 一、摘要1.1 项目介绍1.2 项目录屏 二、系统设计2.1 功能模块设计2.2 研究方法 三、系统…

GridBagLayout GridBagConstraints 笔记231130

实例化使用模板 GridBagLayout gbl new GridBagLayout(); // gbl.columnWidths new int[]{200,200,200}; // 用数组设置列 // gbl.rowHeights new int[]{100,100,100,100,100}; // 用数组设置行GridBagConstraints gbc new GridBagConstraints();/*** gridBagConstrain…

【Spring】Spring统一功能处理

Spring统一功能处理 拦截器拦截器什么是拦截器拦截器的基本使用定义拦截器注册配置拦截器 拦截器详解拦截器的拦截路径配置拦截器实现原理初始化处理请求 适配器模式 统一数据返回格式统一数据返回格式快速入门 统一异常处理 拦截器 场景: 我们要对一个网站实现强制登陆的功能…

TailwindCSS 如何处理RTL布局模式

背景 TikTok作为目前全世界最受欢迎的APP&#xff0c;需要考虑兼容全世界各个地区的本地化语言和阅读习惯。其中对于阿拉伯语、波斯语等语言的阅读书写习惯是从右向左的&#xff0c;在前端有一个专有名字RTL模式&#xff0c;即Right-to-Left。 其中以阿拉伯语作为第一语言的人…

MySQL数据库,创建和管理表

创建数据库&#xff1a; 方式一&#xff1a;创建数据库 CREATE DATABASE 数据库名&#xff1b;&#xff08;使用的是默认的字符集&#xff09; 方式二&#xff1a;创建数据库并指定字符集 CREATE DATABASE 数据库名 CHARACTER SET 字符集&#xff1b; 方式三&#xff1a;判断数…

成都工业学院Web技术基础(WEB)实验三:CSS字体等属性使用

写在前面 1、基于2022级计算机大类实验指导书 2、代码仅提供参考&#xff0c;前端变化比较大&#xff0c;按照要求&#xff0c;只能做到像&#xff0c;不能做到一模一样 3、图片和文字仅为示例&#xff0c;需要自行替换 4、如果代码不满足你的要求&#xff0c;请寻求其他的…

磁学单位SI制和CGS制的转换

电磁学领域中除了使用一般的SI国际制单位外&#xff0c;还会使用CGS高斯制单位&#xff0c;这对于接触磁性材料的朋友们来说&#xff0c;有时就需要做单位的转换&#xff0c;而这两种单位制的转换计算非常复杂。为了方便大家使用&#xff0c;我们系统地总结了一下电磁学中的单位…