【QT5】<总览二> QT信号槽、对象树及常用函数

news2024/11/27 19:43:36

文章目录

前言

一、QT信号与槽

1. 信号槽连接模型

2. 信号槽介绍

3. 自定义信号槽

二、QT的对象树

三、添加资源文件

四、样式表的使用

五、QSS文件的使用

六、常用函数与宏


前言

承接【QT5】<总览一> QT环境搭建、快捷键及编程规范。若存在版权问题,请联系作者删除!


一、QT信号与槽

1. 信号槽连接模型

1.1 信号槽连接模型:

1.2 QT中信号槽的使用方式:

方式一:如下图所示,在ui设计器中的中下位置可以设置信号槽的四要素。下图实现了点击按钮就会关闭当前窗口的功能。

方式二:如下图所示,右击按钮,再点击“转到槽”,QT就会给MyWindow类自动在头文件中声明私有的void on_pushButton_clicked(); 同时也会转到源文件要求程序员实现点击的效果。

2. 信号槽介绍

2.1 概念:

  • 信号:在特定情况下被发射的事件。例如:PushButton最常见的信号就是鼠标单击时发射的click()信号。
  • 槽:对信号响应的函数。槽函数与普通的函数相比,不同在于:槽函数与指定的信号关联,当信号被发射时,该槽函数会被自动执行。

2.2 信号与槽的关联:

  • 方式:使用QObject::connect()函数。由于QT中所有类都是QObject的子类,因此可以省去QObject,其基本格式如下:
//sender 是发射信号的对象的名称
//signal() 是信号名称。信号可以看做是特殊的函数,需要带括号,有参数时还需要指明参数
//receiver 是接收信号的对象名称
//slot() 是槽函数的名称,需要带括号,有参数时还需要指明参数
//SIGNAL和SLOT是Qt的宏,用于指明信号和槽,并将它们的参数转换为相应的字符串
connect(sender, SIGNAL(signal()), receiver, SLOT(slot()));
  • 代码示例:
/* 1.mywindow.h头文件代码 */
#ifndef MYWINDOW_H
#define MYWINDOW_H
#include <QMainWindow>

QT_BEGIN_NAMESPACE
namespace Ui { class MyWindow; }
QT_END_NAMESPACE


class MyWindow : public QMainWindow
{
    Q_OBJECT

public:
    MyWindow(QWidget *parent = nullptr);
    ~MyWindow();

private slots:
    void pushButton_clicked();//自定义槽函数

private:
    Ui::MyWindow *ui;
};
#endif // MYWINDOW_H

/************************************************/

/* 2.mywindow.cpp代码  */
#include "mywindow.h"
#include "ui_mywindow.h"
#include <QDebug>

//MyWindow构造函数的具体实现
MyWindow::MyWindow(QWidget *parent)
    : QMainWindow(parent)
    , ui(new Ui::MyWindow)
{
    ui->setupUi(this);
    connect(ui->pushButton, SIGNAL(clicked()), this, SLOT(pushButton_clicked()));
}

//MyWindow析构函数的具体实现
MyWindow::~MyWindow()
{
    delete ui;
}

//自定义槽函数的实现
void MyWindow::pushButton_clicked()
{
    static int i;
    qDebug() << i++ << " ";
}

2.3 信号与槽的解除连接:

  • disconnect(myObject, 0, 0, 0); 断开一切与 myObject 连接的信号或槽。
  • disconnect(myObject, SIGNAL(mySignal()), 0, 0); 断开所有连接到特定信号的东西。
  • disconnect(myObject, 0, myReceiver, 0); 与指定的接收者断开连接。

2.4 信号槽连接规则:

  • 一个信号可以连接多个槽(触发一个造成多个结果)
  • 多个信号可以连接同一个槽(多个触发源造成一个结果)
  • 一个信号可以连接另一个信号(连锁反应)

3. 自定义信号槽

3.1 自定义信号的语法:

//在类中使用signals关键字
//信号函数只声明不定义,并且返回值为void
signals:
    void sendMessage();//自定义信号函数

3.2 自定义槽的语法:

//使用slots自定义槽
//可以使用C++的三种权限符
//声明后必须给出定义
public slots:
    void goToClass();//自定义的槽函数

//槽函数定义
void Student::goToClass()
{
    qDebug() << "学生上课!" << endl;
}

3.3 案例:

背景:学校发出通知,学生收到后回到班级上课。根据此情景,创建School和Student类,并且在School类中自定义信号“发出通知”,在Student类中自定义槽“去班级”。为了简单演示,我们在MainWindow类中添加School和Studen类对象作为成员变量,在MainWindow的构造函数中创建对象,并构建“学校->通知->学生->上课”的信号槽连接,同时emit发射该信号,来观察实验现象。

①School.h:

#ifndef SCHOOL_H
#define SCHOOL_H

#include <QObject>

class School : public QObject
{
    Q_OBJECT
public:
    explicit School(QObject *parent = nullptr);

signals://自定义信号
    void sendMessage();//学校通知

};

#endif // SCHOOL_H

②Student.h:

#ifndef STUDENT_H
#define STUDENT_H

#include <QObject>

class Student : public QObject
{
    Q_OBJECT
public:
    explicit Student(QObject *parent = nullptr);

signals:

public slots://自定义槽
    void goToClass();

};

#endif // STUDENT_H

③Student.cpp:

#include "student.h"
#include <QDebug>

Student::Student(QObject *parent) : QObject(parent)
{

}

void Student::goToClass()
{
    qDebug() << "学生上课!" << endl;
}

④MainWindow.h:

#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QMainWindow>
#include "school.h"
#include "student.h"

QT_BEGIN_NAMESPACE
namespace Ui { class MainWindow; }
QT_END_NAMESPACE

class MainWindow : public QMainWindow
{
    Q_OBJECT

public:
    MainWindow(QWidget *parent = nullptr);
    ~MainWindow();

private:
    Ui::MainWindow *ui;
    School *school;
    Student *student;
};
#endif // MAINWINDOW_H

⑤MainWindow.cpp:

#include "mainwindow.h"
#include "ui_mainwindow.h"

MainWindow::MainWindow(QWidget *parent)
    : QMainWindow(parent)
    , ui(new Ui::MainWindow)
{
    ui->setupUi(this);
    school = new School(this);
    student = new Student(this);
    connect(school, SIGNAL(sendMessage()), student, SLOT(goToClass()));//信号与槽的连接
    emit school->sendMessage();//发射信号
}

MainWindow::~MainWindow()
{
    delete ui;
    delete school;
    delete student;
}


二、QT的对象树

1. 为什么需要设置父对象?

答:因为需要将某些控件附着于父对象,以便能跟着父对象一起移动,方便控件的管理。

2. 如何设置父对象?

答:有构造函数传参和调用setparent()两种方式,如下所示:

//方式一:构造函数传入父对象地址
pushButton = new QPushButton(this);
//方式二:调用setParent函数
pushButton->setParent(this);

3. QT的对象树机制

答:父对象在调用析构函数前,会将附着于其上面所有的对象析构,再调用自己的析构函数。因此,可以不用显式地delete某些附着的对象,QT的对象树机制会自动将其销毁。


三、添加资源文件

有时需要外部的图片等资源,因此需要将资源文件添加进QT。

步骤一:将资源文件放置于当前工程目录下。

步骤二:添加"QT Resource File":

步骤三:添加资源路径的目录:

步骤四:添加外部图片:


四、样式表的使用

上一章已经加载了外部图片资源文件,本章改变样式表(stylesheet)将该图片呈现于窗口中。若不知道如何编写样式表的代码,可以去“帮助”中查找“stylesheet”查看“Qt Style Sheets Reference ”。

步骤一:在ui设计器中将QLable控件拖拽至窗口中:

步骤二:右击该控件,选中"改变样式表",添加图片:


五、QSS文件的使用

在上一章中,我们利用ui设计器来改变样式表从而加载外部图片。相应地,我们可以通过写代码的方式来改变样式表从而改变某些控件的外观。但是,当样式表书写过多时会影响我们代码的可读性。因此,可以通过qss文件来汇总这些样式表,而不需要在主程序中书写这些样式表。

原始效果:在Widget类的构造函数中写这些样式表。

我们将上述的样式表代码转移到qss文件中,减少主程序代码的复杂度。若不知道怎么写qss样式表,可以参考Qt QSS样式表总结

步骤一:新建qss文件:

步骤二:编辑qss文件:

步骤三:main()中加载qss文件:

运行结果:

/* 加载QSS文件 */
QFile file(":/style.qss");
if (file.exists()){
    file.open(QFile::ReadOnly);//只读方式打开
    QString styleSheet = QLatin1String(file.readAll());//字符串方式保存结果
    qApp->setStyleSheet(styleSheet);//设置全局样式
    file.close();//关闭文件
}

六、常用函数与宏

常用宏如下:

【1】Q_OBJECT:声明在private中,用到了信号槽就要添加。

【2】Q_UNUSED(变量名):告诉编译器不使用某个变量。

【3】Q_CLASSINFO(key, value):给出类的相关信息。例如:Q_CLASSINFO("Author", "swear")

//设置类的信息
Q_CLASSINFO("Author", "swear")
Q_CLASSINFO("version", "1.0")

//循环读取类的信息,需要添加相关的头文件
const QMetaObject *meta = boy->metaObject();
for (int i = 0; i < meta->classInfoCount(); ++i){
   QMetaClassInfo classInfo =  meta->classInfo(i);
   qDebug() << classInfo.name() << classInfo.value();
}

【4】Q_PROPERTY(...):属性宏,用于给用户设置、获取属性。若设置属性不存在时,会设置动态属性。设置属性: "对象指针->setProperty(识别名, 值)"。获取属性: "对象指针->property(识别名).toXxx"。

//假设我们有个QPerson类,类内有属性m_name, m_age, m_score
//通过setProperty设置name时,就会去执行setName
Q_PROPERTY(QString name READ getName WRITE setName)

//以下将"age"与属性m_age绑定,之后的读取都是针对m_age
Q_PROPERTY(unsigned int age MEMBER m_age)
Q_PROPERTY(unsigned int score MEMBER m_score)

//创建对象,读写属性
QPerson p;
p.setProperty("name", "王甩笼");//就会去调用setName方法
p.setProperty("age", 10);//将p内部属性m_age设置为10
qDebug() << p.property("name").toString();//打印"王甩笼"

常用函数如下:

【1】QWidget类中的update():更新界面。

【2】QWidget类中的show():显示界面。若某个界面类在实例化时没有指定父对象,则需要调用该函数来显示界面。

【3】sender():在槽函数中获取发出信号的对象,用指针变量接收。例如QTcpSocket *tmpTcpSocket = (QTcpSocket*)sender();

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Python使用tkinter库设置背景图片、label显示位置和label设置显示图片

tkinter 设置背景图片 label显示位置 label设置显示图片 from tkinter import * import tkinter as tk from PIL import ImageTk from PIL import Imagedef get_img(filename, width, height):im Image.open(filename).resize((width, height))im ImageTk.PhotoImage(im)…

网络仿真方法综述

目录 1. 引言 2.仿真器介绍 2.1 NS-2 2.2 NS-3 2.3 OPNET 2.4 GNS3 3.仿真对比 4.结论 参考文献 1. 引言 网络仿真是指使用计算机模拟网络系统的行为和性能的过程。在网络仿真中&#xff0c;可以建立一个虚拟的网络环境&#xff0c;并通过模拟各种网络设备、协议和应用程…

this关键字,构造函数(构造器)

文章目录 thisthis是什么应用场景 构造器注意事项代码演示 this this是什么 this就是一个变量&#xff0c;可以在方法中&#xff0c;拿到当前对象 应用场景 解决变量名称 冲突问题 构造器 注意事项 必须和类名相同没有返回值只要参数不同&#xff08;个数不同&#xff0…

【C语言】10.操作符详解

一、操作符分类 • 算术操作符&#xff1a; 、- 、* 、/ 、% • 移位操作符: << 、 >> • 位操作符: & 、|、^、 ~ • 赋值操作符: 、 、 - 、 、 / 、% 、<< 、>> 、& 、| 、^ • 单目操作符&#xff1a; &#xff01;、、–、&、、、…

postman教程-20-Newman安装入门

上一小节我们学习了Postman mock测试的方法&#xff0c;本小节我们讲解一下Postman Newman的安装方法。 Newman是Postman团队开发的一个命令行工具&#xff0c;它允许用户通过命令行接口&#xff08;CLI&#xff09;运行Postman集合&#xff08;Collections&#xff09;和环境…

样式的双向绑定的2种方式,实现样式交互效果

与样式标签实现双向绑定 通过布尔值来决定样式是出现还是消失 show代表着布尔值&#xff0c;show的初始值是false所以文本不会有高亮的效果&#xff0c;当用户点击了按钮&#xff0c;就会调用shows这个函数&#xff0c;并将show的相反值true赋值并覆盖给show,此时show的值为tru…

0602 差分式放大电路

差分式放大电路 差分放大电路的基本概念直接耦合放大电路中的零点漂移 6.2.1 差分式放大的基本概念 6.2.2 直接耦合放大电路中的零点漂移 6.2.3 BJT射极耦合差分式放大电路 差分放大电路的基本概念 直接耦合放大电路中的零点漂移

.NET Core 服务注册步骤总结

总结一下 .NET Core 服务注册的步骤&#xff1a; .NET Core Web Api 项目服务注册步骤&#xff1a; 创建一个接口&#xff0c;和实现类 比如&#xff1a;IMyService, CnService 在 Program.cs 的 var app builder.Build(); 语句之前加上&#xff1a; var builder WebApplic…

鸿蒙开发:【线程模型】

线程模型 线程类型 Stage模型下的线程主要有如下三类&#xff1a; 主线程 执行UI绘制。管理主线程的ArkTS引擎实例&#xff0c;使多个UIAbility组件能够运行在其之上。管理其他线程的ArkTS引擎实例&#xff0c;例如使用TaskPool&#xff08;任务池&#xff09;创建任务或取消…

使用 PNPM 从 0 搭建 monorepo,测试并发布

1 目标 通过 PNPM 创建一个 monorepo&#xff08;多个项目在一个代码仓库&#xff09;项目&#xff0c;形成一个通用的仓库模板。 这个仓库既可以用于公司存放和管理所有的项目&#xff0c;也可以用于将个人班余的所有积累整合其中。 2 环境要求 核心是 PNPM 和 Node.js&…

SpringAI调用OpenAI Demo

Spring AI 在maven的setting.xml <mirror> <id>spring-milestones</id> <name>Spring Milestones</name> <mirrorOf>spring-milestones</mirrorOf> <url>https://repo.sprin…

RV32A\CSR\Counters 指令集

RV32A\CSR\Counters指令集 一、RV32A指令集1、Load-Reserved/Store-Conditional InstructionsLR.WSC.W2、Atomic Memory OperationsAMOSWAP.WAMOADD.WAMOAND.WAMOXOR.WAMOOR.W二、CSR(Control and Status Register) 指令集CSRRWCSRRSCSRRCCSRRWICSRRSICSRRCI三、"Zicntr…

uniapp上传头像并裁剪图片

第一步写上uniapp自带的选择图片button按钮 点击之后会弹出选择图片的方式 拍照或从相册选择图片后将会跳到图片裁剪 然后我们裁剪完之后点击确定在上传图片 这里是上传图片的接口 拿到本地图片 上传的话自己想以那种方式上传都可以

Multimodal Dynamics:用于多模态融合背景下的分类

Multimodal Dynamics&#xff08;MD&#xff09;是可信赖的多模态分类算法&#xff0c;该算法动态评估不同样本的特征级和模态级信息量&#xff0c;从而可信赖地对多模态进行融合。 来自&#xff1a;Multimodal Dynamics: Dynamical Fusion for Trustworthy Multimodal Classi…

代理模式与静态代理、动态代理的实现(Proxy.newProxyInstance、InvocationHandler)

代理模式 代理模式是23种设计模式中比较常用的一种&#xff0c;属于结构型设计模式。在 Android 领域中&#xff0c;有大量的库都使用了代理模式&#xff0c;例如 Retrofit 使用动态代理来实现 API 接口的调用&#xff0c;Dagger 使用代码生成和反射机制来创建依赖注入的代理对…

Bybatis动态SQL的绑定和公共sql语句片段

Mybatis除了大部分动态标签&#xff0c;最后还有三个标签&#xff0c;分别是bind&#xff0c;sql和include&#xff1a; ①bind&#xff1a;这个标签作用就是将OGNL标签里的值&#xff0c;进行二次加工&#xff0c;在绑定到另一个变量里&#xff0c;供其他标签使用 调用getUse…

RocketMQ集群搭建(1)

1.1 各角色介绍 Producer&#xff1a;消息的发送者&#xff1b;举例&#xff1a;发信者Consumer&#xff1a;消息接收者&#xff1b;举例&#xff1a;收信者Broker&#xff1a;暂存和传输消息&#xff1b;举例&#xff1a;邮局NameServer&#xff1a;管理Broker&#xff1b;举…

【CH32V305FBP6】USBD HS 描述符修改

文章目录 前言设备描述符完整描述符配置描述符CDC 描述符接口关联描述符接口描述符功能描述符端点描述符接口描述符端点描述符 HID 描述符接口描述符 新增一个 HID 设备 前言 USB HS 复合设备&#xff0c;CDCHID 功能&#xff1a;串口、DAP、CAN-HID、RS485 设备描述符 htt…

DockerCompose+Jenkins+Pipeline流水线打包SpringBoot项目(解压安装配置JDK、Maven等)入门

场景 DockerCompose中部署Jenkins&#xff08;Docker Desktop在windows上数据卷映射&#xff09;&#xff1a; DockerCompose中部署Jenkins&#xff08;Docker Desktop在windows上数据卷映射&#xff09;-CSDN博客 DockerJenkinsGiteeMaven项目配置jdk、maven、gitee等拉取代…

[12] 使用 CUDA 进行图像处理

使用 CUDA 进行图像处理 当下生活在高清摄像头的时代&#xff0c;这种摄像头能捕获高达1920*1920像素的高解析度画幅。想要实施的处理这么多的数据&#xff0c;往往需要几个TFlops地浮点处理性能&#xff0c;这些要求CPU也无法满足通过在代码中使用CUDA&#xff0c;可以利用GP…