QT_02 窗口属性、信号槽机制

QT - 窗口属性、信号槽机制

1. 设置窗口属性

窗口设置

1,标题
2,大小
3,固定大小
4,设置图标

widget.cpp 文件中:

//设置窗口大小,此时窗口是可以拉大拉小的
//1参:宽度
//2参:高度
this->resize(800, 600);
//设置窗口标题
this->setWindowTitle("QT第一个窗口界面");
//设置窗口大小不可改变
this->setFixedSize(800, 600);
//设置图标
//this->setWindowIcon();

在这里插入图片描述

2. 按钮 QPushButton

构造函数QPushButton(父容器)设置文本setText
获取文本text
设置大小resize
移动move

2.1 方式一:代码创建

widget.cpp 文件中:

#include "widget.h"
#include "ui_widget.h"
#include <QPushButton>  //1、引入按钮所需头文件
Widget::Widget(QWidget *parent) :QWidget(parent),ui(new Ui::Widget)
{ui->setupUi(this);ui->setupUi(this);//设置窗口大小,此时窗口是可以拉大拉小的//1参:宽度//2参:高度this->resize(800, 600);//设置窗口标题this->setWindowTitle("QT第一个窗口界面");//设置窗口大小不可改变this->setFixedSize(800, 600);//2、创建按钮对象QPushButton *btn01 = new QPushButton;//3、设置按钮父容器,此处是当前窗口btn01->setParent(this);//3.1 设置按钮大小//btn01->setFixedSize(300,50);//4、设置位置btn01->move(100, 100);//5、设置按钮文本btn01->setText("按钮");
}Widget::~Widget()
{delete ui;
}

在这里插入图片描述

2.2 方式二:图形界面

在这里插入图片描述

在这里插入图片描述

3. 信号与槽机制

//Qt4: 
connect(btn, SIGNAL(clicked(bool)), this, SLOT( close() ) );
//Qt5: 
connect(btn, &QPushButton::clicked, this, &QWidget::close );

3.1 概述

信号槽是 Qt 框架引以为豪的机制之一。

所谓信号槽,实际就是 观察者模式

  • 当某个事件发生之后,比如,按钮检测到自己被点击了一下, 它就会发出一个信号( signal) 。这种发出是没有目的的,类似广播。 如果有对象对这个信号感兴趣,它就会使用连接( connect)函数,意思是, 将想要处理的信号和自己的一个函数(称为槽( slot))绑定来处理这个信号。
  • 也就是说, 当信号发出时, 被连接的槽函数会自动被回调。这就类似观察者模式:当发生了感兴趣的事件,某一个操作就会被自动触发。

在这里插入图片描述

注意:

  • 核心在于 发送者 与 接受者
  • 发送者 可以 发出多种信号,被多个不同的接收者接收
  • 接收者 有多个槽函数接收不同的发送者发出的信号

3.2 信号与槽的链接

connect() 函数

connect(sender, signal, receiver, slot)

参数:

  • sender:发送者
  • signal:发出的信号
  • receiver:接受者
  • slot:槽函数

示例1:

//ui文件中名为btnclose的按钮发出点击信号
//被当前窗口接收,执行关闭窗口的操作//qt5的写法
connect(ui->btnclose,&QPushButton::clicked,this,&Widget::close);//qt4的写法
//connect(ui->btnclose,SIGNAL(clicked(bool)),this,SLOT(close()));

示例2:

//信号与槽函数有参数
//qt5的写法
//void (QPushButton:: *cli_p)(bool) = &QPushButton::clicked;
//void (Widget:: *myfun_p)(bool) = &Widget::myfun;
//connect(ui->btnclose,cli_p,this,myfun_p);
//qt4的写法
connect(ui->btnclose,SIGNAL(clicked(bool)),this,SLOT(myfun(bool)));

3.3 系统提供的信号与槽

3.3.1 QWidget提供

信号:

void customContextMenuRequested(const QPoint &pos) //请求上下文菜单时
void windowIconChanged(const QIcon &icon) //窗口图标改变时
void windowTitleChanged(const QString &title) //窗口标题改变时

槽:

bool close() // 关闭
void hide() // 隐藏
void lower()
void raise()
void repaint() // 重新加载
void setDisabled(bool disable)
void setEnabled(bool)
void setFocus()
void setHidden(bool hidden)
void setStyleSheet(const QString &styleSheet)
virtual void setVisible(bool visible)
void setWindowModified(bool)
void setWindowTitle(const QString &)
void show() // 显示
void showFullScreen() // 全屏显示
void showMaximized() // 最大化显示
void showMinimized() // 最小化显示
void showNormal()
void update()
3.3.2 QPushButton提供

继承于父类的 QAbstractButton的信号

信号:

void clicked(bool checked = false) //点击信号
void pressed() //按钮按下信号 (按下)
void released() //按钮释放信号(抬起)
void toggled(bool checked) //触发(开或关)
3.3.3 示例

如1: 当前窗口添加一个关闭按钮,点击之后关闭窗口(退出程序)

#include "widget.h"
#include <QPushButton>
Widget::Widget(QWidget *parent): QWidget(parent)
{setFixedSize(800, 640);QPushButton *btn= new QPushButton("关闭", this);btn->setFixedSize(120, 50);btn->move(10, 10);// 当前类对象对 QPushButton的点击事件感兴趣// 使用connect()进行绑定到当前窗口的close()// 发送者和接收者都是QObject类的对象的指针// Qt5的信号绑定槽函数的方式connect(btn, &QPushButton::clicked, this, &Widget::close);
}

如2:当前窗口中添加一个按钮,当按下时最大化显示窗口,再点时,恢复之前的状态

#include "widget.h"
#include <QPushButton>
#include <QDebug>
Widget::Widget(QWidget *parent): QWidget(parent)
{resize(800, 640); // 窗口的初始大小// setMaximumSize(1200, 960); // 设置窗口最大值QPushButton *btn= new QPushButton("关闭", this);btn->setFixedSize(120, 50);btn->move(10, 10);// 当前类对象对 QPushButton的点击事件感兴趣// 使用connect()进行绑定到当前窗口的close()// 发送者和接收者都是QObject类的对象的指针// Qt5的信号绑定槽函数的方式connect(btn, &QPushButton::clicked, this, &Widget::close);QPushButton *maxBtn = new QPushButton("最大化", this);maxBtn->setFixedSize(120, 50);maxBtn->move(10, 70);// 绑定的槽函数是自定义的成员函数connect(maxBtn, &QPushButton::clicked, this, &Widget::toggleShow);
}// 在public区域声明的函数 toggleShow()
void Widget::toggleShow(){// qDebug()引入 <QDebug> 头qDebug() << "show or hide:" << this->isMaximized() << endl;if(isMaximized()){showNormal(); // 槽函数可以作为成员函数使用}else{showMaximized();}
}

3.4 自定义信号与槽

3.4.1 注意事项
  1. 发送者和接收者都需要是QObject的子类(当然,槽函数是全局函数、Lambda 表达式等无需接收者的时候除外)
  2. 信号和槽函数 返回值类型是 void
  3. 信号 只需要声明,不需要实现
  4. 槽函数 需要声明也需要实现
  5. 槽函数是普通的成员函数,作为成员函数,会受到 public、private、protected的影响;
  6. 使用 emit 在恰当的位置发送信号;
  7. 使用 connect() 函数 连接信号和槽。
  8. 任何成员函数、static 函数、全局函数和 Lambda 表达式都可以作为槽函数
  9. 信号槽要求信号和槽的 参数一致,所谓一致,是 参数类型一致
  10. 如果信号和槽的参数不一致,允许的情况是,槽函数的参数可以比信号的少,即便如此,槽函数存在的那些参数的顺序也必须和信号的前面几个一致起来。这是因为,你可以在槽函数中选择忽略信号传来的数据(也就是槽函数的参数比信号的少)。
3.4.2 无参

下课了,老师饿了,学生请吃饭

widget.h 没变化

widget.cpp

#include "widget.h"
#include "ui_widget.h"
#include "teacher.h"
#include "student.h"
#include <QPushButton>Widget::Widget(QWidget *parent) :QWidget(parent),ui(new Ui::Widget)
{ui->setupUi(this);this->resize(800, 600);this->setFixedSize(800, 600);this->setWindowTitle("老师饿了,学生请吃饭");QPushButton *btn = new QPushButton;btn->setParent(this);btn->move(100, 100);btn->setText("下课");//创建学生和老师对象Teacher *tea = new Teacher();Student *stu = new Student();//关联1:按钮和老师对象,//按钮:发送者; clicked:发出的信号; tea:接收者; down:槽函数connect(btn, QPushButton::clicked, tea, Teacher::down);//关联2:老师和学生对象,//tea:发送者; hungry:发出的信号; stu:接收者; eat:槽函数connect(tea, Teacher::hungry, stu, Student::eat);
}Widget::~Widget()
{delete ui;
}

新建 teacher.h

#ifndef TEACHER_H
#define TEACHER_H#include <QObject>class Teacher : public QObject
{Q_OBJECT
public:explicit Teacher(QObject *parent = 0);signals://信号void hungry();public slots://槽函数void down();
};#endif // TEACHER_H

新建 teacher.cpp

#include "teacher.h"
#include <QDebug>Teacher::Teacher(QObject *parent) : QObject(parent)
{}void Teacher::down()
{//发送信号emit this->hungry();
}

新建 student.h

#ifndef STUDENT_H
#define STUDENT_H#include <QObject>class Student : public QObject
{Q_OBJECT
public:explicit Student(QObject *parent = 0);signals:public slots:void eat();
};#endif // STUDENT_H

新建 student.cpp

#include "student.h"
#include <QDebug>Student::Student(QObject *parent) : QObject(parent)
{}void Student::eat()
{qDebug() << endl << "请吃饭" << endl;
}

在这里插入图片描述

3.4.3 有参

widget.cpp

#include "widget.h"
#include "ui_widget.h"
#include "teacher.h"
#include "student.h"
#include <QPushButton>Widget::Widget(QWidget *parent) :QWidget(parent),ui(new Ui::Widget)
{ui->setupUi(this);this->resize(800, 600);this->setFixedSize(800, 600);this->setWindowTitle("老师饿了,学生请吃饭");QPushButton *btn = new QPushButton;btn->setParent(this);btn->move(100, 100);btn->setText("下课");Teacher *tea = new Teacher();Student *stu = new Student();//关联1:按钮和老师对象,//按钮:发送者; clicked:发出的信号; tea:接收者; down:槽函数connect(btn, QPushButton::clicked, tea, Teacher::down);//关联2:老师和学生对象,//tea:发送者; hungry:发出的信号; stu:接收者; eat:槽函数
//    connect(tea, Teacher::hungry, stu, Student::eat01);connect(tea, Teacher::hungry, stu, Student::eat02);//槽的参数只能小于等于信号的参数,eat03报错connect(tea, Teacher::hungry, stu, Student::eat03);
}Widget::~Widget()
{delete ui;
}

新建 teacher.h

#ifndef TEACHER_H
#define TEACHER_H#include <QObject>
#include <QString>class Teacher : public QObject
{Q_OBJECT
public:explicit Teacher(QObject *parent = 0);signals://有参数void hungry(int num, QString foodName);public slots:void down();
};#endif // TEACHER_H

新建 teacher.cpp

#include "teacher.h"
#include <QDebug>Teacher::Teacher(QObject *parent) : QObject(parent)
{}void Teacher::down()
{emit this->hungry(3, "油泼面");
}

新建 student.h

#ifndef STUDENT_H
#define STUDENT_H#include <QObject>class Student : public QObject
{Q_OBJECT
public:explicit Student(QObject *parent = 0);signals:public slots://小于信号参数void eat01();//大于信号参数void eat02(int num, QString foodName);//大于信号参数void eat03(int num, int x, QString foodName);
};#endif // STUDENT_H

新建 student.cpp

#include "student.h"
#include <QDebug>Student::Student(QObject *parent) : QObject(parent)
{}void Student::eat01()
{qDebug() << endl << "eat01" << endl;
}void Student::eat02(int num, QString foodName)
{qDebug() << endl << "eat02:\tnum:" << num << "\tfoodName:" << foodName << endl;
}void Student::eat03(int num, int x, QString foodName)
{qDebug() << endl << "eat03" << endl;
}

在这里插入图片描述
在这里插入图片描述

3.5 信号与槽拓展

3.5.1 一个信号可以和多个槽相连

槽会一个接一个的被调用,但是它们的调用顺序是不确定

3.5.2 多个信号可以连接到一个槽

只要任意一个信号发出,这个槽就会被调用

3.5.3 一个信号可以连接到另外的一个信号

当第一个信号发出时,第二个信号被发出。除此之外,这种信号-信号的形式和信号-槽的形式没有什么区别。

3.5.4 信号槽可以断开

利用 disconnect 关键字是可以断开信号槽的

3.5.5 槽可以被取消链接

这种情况并不经常出现,因为当一个对象 delete 之后,Qt 自动取消所有连接到这个对象上面的槽

3.5.6 示例

widget.h

#ifndef WIDGET_H
#define WIDGET_H#include <QWidget>namespace Ui {
class Widget;
}class Widget : public QWidget
{Q_OBJECTpublic:explicit Widget(QWidget *parent = 0);~Widget();private:Ui::Widget *ui;public slots:void myslot01();void myslot02();void myslot03();
};#endif // WIDGET_H

widget.cpp

#include "widget.h"
#include "ui_widget.h"
#include <QPushButton>
#include <QDebug>Widget::Widget(QWidget *parent) :QWidget(parent),ui(new Ui::Widget)
{ui->setupUi(this);setWindowTitle("信号与槽的拓展");setFixedSize(800,600);QPushButton *btn01 = new QPushButton("一个信号连接多个槽",this);btn01->resize(300,50);connect(btn01,QPushButton::clicked,this,Widget::myslot01);connect(btn01,QPushButton::clicked,this,Widget::myslot02);connect(btn01,QPushButton::clicked,this,Widget::myslot03);QPushButton *btn02 = new QPushButton("多个信号连接一个槽",this);btn02->resize(300,50);btn02->move(0,50);connect(btn02,QPushButton::clicked,this,Widget::myslot01);QPushButton *btn03 = new QPushButton("一个信号连接另一个信号",this);btn03->resize(300,50);btn03->move(0,100);connect(btn03,QPushButton::clicked,btn01,QPushButton::clicked);//槽可以被取消链接//delete btn01;//信号与槽可以断开
//    btn03->disconnect(btn01);
}Widget::~Widget()
{delete ui;
}void Widget::myslot01()
{qDebug() << endl << "myslot01" << endl;
}
void Widget::myslot02()
{qDebug() << endl << "myslot02" << endl;
}
void Widget::myslot03()
{qDebug() << endl << "myslot03" << endl;
}

在这里插入图片描述

  • 一个信号连接多个槽:

在这里插入图片描述

  • 多个信号连接一个槽

    在这里插入图片描述

  • 一个信号连接另一个信号

    在这里插入图片描述

  • 信号与槽可以断开:断开后无响应

3.6 Lambda表达式

C++11 中的 Lambda 表达式用于定义并创建匿名的函数对象

作用:简化编程工作。

3.6.1 语法
[函数对象参数](操作符重载函数参数) mutable ->返回值{函数体}

解释:

  • [ ]:lambda表达式符号,,可以啥都不写,不能省略
    • 函数对象参数
      • 空 没有使用任何函数对象参数
      • = 可以访问外部变量只能读
      • a,b 能对lambda外的a,b变量读操作
      • & lambda外的变量读写操作
      • a,&b 对外部的a读 b读写
      • this 函数体内可以使用 Lambda 所在类中的成员变量
  • ():形参列表
3.6.2 示例

widget.h

#ifndef WIDGET_H
#define WIDGET_H#include <QWidget>namespace Ui {
class Widget;
}class Widget : public QWidget
{Q_OBJECTpublic:explicit Widget(QWidget *parent = 0);~Widget();private:Ui::Widget *ui;int m;int n;
public slots:myslot01();
};#endif // WIDGET_H

widget.cpp

#include "widget.h"
#include "ui_widget.h"
#include <QPushButton>
#include <QDebug>Widget::Widget(QWidget *parent) :QWidget(parent),ui(new Ui::Widget)
{ui->setupUi(this);setWindowTitle("QT4连接信号与槽的写法");setFixedSize(800,600);QPushButton *btn = new QPushButton("QT4",this);btn->resize(300,50);//QT4写法//connect(btn, SIGNAL(clicked(bool)), this, SLOT(myslot01()));//QT5的写法//connect(btn,QPushButton::clicked,this,Widget::myslot01);//lambda表达式//语法:[...](形参列表){函数体};
//    connect(btn,QPushButton::clicked,[](){
//        qDebug() << "Lambda函数" << endl;
//    });//    int num = 10;
//    //=,可以访问外部变量,但是不能修改
//    connect(btn,QPushButton::clicked,[=](){
//        qDebug() << "Lambda函数" << num << endl;
//    });//    int x = 11;
//    int y = 22;
//    //能对lambda外的x,y变量只读操作,修改会报错
//    connect(btn,QPushButton::clicked, [x, y](){
//        qDebug() << "Lambda函数" << x << endl;
//        qDebug() << "Lambda函数" << y << endl;
//    });//局部变量,可以修改, 访问是随机值,因为这块代码结束局部变量会弹栈销毁,//所以要声明成全局变量或者用static修饰//此处为全局变量,访问全局变量用this,用static修饰时,[]中为[&m, &n]即可。m = 11;n = 22;connect(btn,QPushButton::clicked,[this](){qDebug() << "Lambda函数" << m << endl;qDebug() << "Lambda函数" << n << endl;m = 111;n = 222;qDebug() << "Lambda函数" << n << endl;qDebug() << "Lambda函数" << n << endl;});
}Widget::~Widget()
{delete ui;
}Widget::myslot01()
{qDebug() << "slot01" << endl;
}

结果:

在这里插入图片描述

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

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

相关文章

出个花活:出街&秀场丨当维乐VELO遇上英伦时尚之都

到底是谁还没有看过我们维乐坐垫今年的新花活呀&#xff0c;身边好多从前不爱运动的朋友&#xff0c;如今也沉迷上了公路车。我相信原因一定是由于对产品设计有着更高的要求&#xff0c;对于审美有着越来越高的追求&#xff0c;也是因为此大多数朋友最终都选择了维乐专业坐垫&a…

自定义页面,落地页面自由搭配

自定义页面 路径 应用 >> 新增自定义页面 功能简介 应用内新增「自定义页面」。 自定义页面是一个可以自由配置的落地页面&#xff0c;支持通过不同的入口设置连接到不同的链接地址&#xff0c;使得不同的应用资源可以根据业务场景化的展示。 使用场景&#xff1a; 一…

打造强大的Android C++工程

theme: cyanosis 好久没有写博客了&#xff0c;最近一直在做项目重构&#xff0c;动刀之深&#xff0c;让我无暇其他。今天终于告一段落了&#xff0c;就总结一下前段时间学习C时的一些开发心得吧。 因为Android系统&#x1f236;️C 语言开发的原因&#xff0c;每个 Android…

Flutter 混合开发 - aar打包

背景 项目接入 Flutter 后有两种方式&#xff0c;一种是 module 引入开发&#xff0c;一种是 aar 依赖开发。当前项目中在 Debug 阶段为了方便调试采用 module 开发&#xff0c;在发版时&#xff08;即 Release 阶段&#xff09;采用 aar 依赖引入。为了配合这种模式就需要在 …

【2023年度技术盘点】「年终盘点后端系列」探索服务架构体系的技术风向,构建微服务核心能力(升级版)

探索服务架构体系的技术风向&#xff0c;构建微服务核心能力 文章导航大纲前提背景架构未来的风向云原生化的微服务架构&#xff08;未来软件架构&#xff09;历史历代服务架构路径新时代架构预测服务架构方向—云原生化微服务云原生化微服务提升了哪些方面 云原生化微服务架构…

XV4001BC (数字输出) 车载用

XV4001BC、XV4001BD、XV4001KC和XV4001KD是车载用SPI/I2C输出接口&#xff0c;支持角速度输出、温度传感器输出以及20度倾斜实现。这些接口可适应各种汽车导航和通讯系统&#xff0c;符合AEC-Q200标准 SPI / I2C 输出接口角速度输出(16bit)、温度传感器输出(11bit)20度倾斜实现…

OpenAI ChatGPT-4开发笔记2024-04:Chat之Tool之2:multiple functions

从程序员到ai Expert 1 设定目标2 自定义function,3个3 接口。自定义function--->ChatGPT4 define function to call ChatGPT5 发起首次请求&#xff0c;告诉gpt要做什么&#xff0c;已经有哪些函数可以调动6 大结局7 参考资料 上一篇解决了调用一个函数的问题。这一篇扩展为…

钉钉审批流程解读

组织机构 部门 部门可以创建下级部门部门可以设置部门主管&#xff0c;可以是多人部门可以默认构建&#xff0c;沟通群可以设置部门信息&#xff0c;比如电话、简介可以设置部门的可见性&#xff0c;比如隐藏本部门&#xff0c;本部门将不会在组织机构、搜索&#xff0c;个人…

Activiti7官方在线流程设计器下载和部署

文章目录 一、流程设计器下载二、流程设计器简单运行三、流程设计器简单使用四、流程设计器持久化持久化会遇到的常见错误 五、流程设计器汉化说明菜单汉化操作汉化 参考文档 一、流程设计器下载 官网下载地址&#xff1a;https://www.activiti.org/get-started 点击直接获取官…

1.1 Heterogeneous Parallel Computing

1.1 Heterogeneous Parallel Computing 前言HETEROGENEOUS PARALLEL COMPUTING 前言 基于单个中央处理器&#xff08;CPU&#xff09;的微处理器&#xff0c;如英特尔奔腾系列和AMD皓龙系列的微处理器&#xff0c;二十多年来推动了计算机应用程序的性能快速提高和成本降低。 …

Qt中图片旋转缩放操作

在我们开发过程中&#xff0c;难免会遇到加载图片的问题&#xff0c;在上一个开发项目里我就遇到了图片缩放的问题&#xff0c;所以&#xff0c;我决定将这一部分好好研究&#xff0c;记录下来&#xff0c;希望对大家有帮助哟~ 在讲解之前&#xff0c;我们先看一看具体的展示效…

运行了一晚上的经典京东面试原题

写在前面 今天在「京东」题库中翻到一道经典题。 众所周知&#xff0c;题目越经典&#xff0c;评论区越逆天。 可能是的呢 &#x1f923; 评论区常规操作 怀疑人生第一步&#xff1a;做道经典题 小白初评&#xff1f; 说归说&#xff0c;闹归闹&#xff0c;这道题还是要掌握的。…