QT基础入门——信号和槽机制(二)

简介: QT基础入门——信号和槽机制(二)

一、QPushButton Class的创建

按钮控件头文件:#include

#include "mywidget.h"
#include<QPushButton>
 
 
 
myWidget::myWidget(QWidget *parent)
    : QWidget(parent)
{
    //创建一个按钮
    QPushButton * btn = new QPushButton;
    //btn->show();//show以顶层方式弹出窗口控件
    //让btn对象  依赖在myWidget窗口中
    btn->setParent(this);
 
    //显示文本
    btn->setText("第一个按钮");
 
}
 
myWidget::~myWidget()
{
}
 

#include "mywidget.h"
#include<QPushButton>
 
 
 
myWidget::myWidget(QWidget *parent)
    : QWidget(parent)
{
   
    //创建第二个按钮,按照控件大小创建窗口
    QPushButton * btn2 = new QPushButton("第二个按钮",this);
 
    //移动btn2按钮
    btn2->move(100,100);
 
    //重置窗口大小
    resize(600,400);
 
    //设置固定窗口大小
    setFixedSize(600,400);
 
    //设置窗口标题
    setWindowTitle("窗口");
}
 
myWidget::~myWidget()
{
}
 

二、对象树

当创建的对象在堆区时候,如果指定的父亲是object派生下来的类或者.QObject子类派生下来的类,可以不用管理释放的操作,将对象会放入到对象树中。一定程度上简化了内存回收机制

 

1.创建:

2. mypushbutton.h

#ifndef MYPUSHBUTTON_H
#define MYPUSHBUTTON_H
 
#include <QPushButton>
 
class MyPushButton : public QPushButton
{
    Q_OBJECT
public:
    explicit MyPushButton(QWidget *parent = nullptr);
 
    ~MyPushButton();
signals:
 
};
 
#endif // MYPUSHBUTTON_H

3. mypushbutton.cpp

#include "mypushbutton.h"
#include <QDebug>
 
MyPushButton::MyPushButton(QWidget *parent) : QPushButton(parent)
{
    qDebug() << "我的按钮类构造调用";
}
 
MyPushButton ::~MyPushButton()
{
    qDebug() << "我的按钮类析构";
}

4.mywidget.cpp

#include "mywidget.h"
#include <QPushButton>
#include <mypushbutton.h>
#include <QDebug>
 
myWidget::myWidget(QWidget *parent)
    : QWidget(parent)
{
 
    //创建一个自己按钮的对象
    MyPushButton * myBtn = new MyPushButton;
    myBtn->setText("我自己的按钮");
 
    myBtn->move(200,0);
    myBtn->setParent(this); //设置到对象树中
}
 
myWidget::~myWidget()
{
    qDebug() << "myWidget的析构调用";
}
 

5.运行结果

三、QT坐标系

坐标系:

以左上角为原点(0,0)X向右增加   Y向下增加

其坐标相对于父窗口而言

*四.信号和槽

connect(信号的发送者,发送的具体信号,信号的接受者,信号的处理(槽))

信号槽的优点:

松散耦合﹐信号发送端和接受端本身是没有关联的,通过connect连接将两端耦合在一起

1.点击按钮关闭窗口

参数1   信号的发送者  

参数2   发送的信号(函数的地址)  

参数3   信号的接受者    

参数4   处理的槽函数(函数的地址)

#include "mywidget.h"
#include <QPushButton>
#include <mypushbutton.h>
#include <QDebug>
 
myWidget::myWidget(QWidget *parent)
    : QWidget(parent)
{
 
    //创建一个自己按钮的对象
    MyPushButton * myBtn = new MyPushButton;
    myBtn->setText("我自己的按钮");
 
    myBtn->move(200,0);
    myBtn->setParent(this); //设置到对象树中
 
    //需求   点击我的按钮关闭窗口
    //参数1  信号的发送者   参数2   发送的信号(函数的地址)   参数3   信号的接受者    参数4    处理的槽函数
    //connect(myBtn,&MyPushButton::clicked,this,&myWidget::close);
    connect(myBtn,&QPushButton::clicked,this,&QWidget::close);
}
 
myWidget::~myWidget()
{
    qDebug() << "myWidget的析构调用";
}
 

2.自定义信号和槽

(1)自定义信号

       写到signals下

       返回void

       需要声明,不需要实现

       可以有参数,可以重载

(2)自定义槽函数

       返回void

       需要声明,也需要实现

       可以有参数,可以重载

       写到public slot下或者public或者全局函数。

(3)当定义信号和槽出现重载

       需要利用函数指针明确指向函数地址

void(Teacher:: *teacherSignal)(QString) = &Teacher::hungry;    


   void(Student:: *studentSlot)(QString) = &Student::treat;  


   connect(zt,teacherSignal,st,studentSlot);

       QsString ----》char *先转成OByteArray(.toUtf8())再转char *()

(4)信号连接信号

  无参信号和槽的连接  

   void(Teacher:: *teacherSignal2)(void) = &Teacher::hungry;  

 

   void(Student:: *studentSlot2)(void) = &Student::treat;    

       

   connect(zt,teacherSignal2,st,studentSlot2);  

   信号连接信号  

   connect(btn,&QPushButton::clicked,zt,teacherSignal2);

(5)断开信号  

disconnect(zt,teacherSignal2,st,studentSlot2);

3.代码实现

(1)student.h

#ifndef STUDENT_H
#define STUDENT_H
 
#include <QWidget>
 
class Student : public QWidget
{
    Q_OBJECT
public:
    explicit Student(QWidget *parent = nullptr);
 
signals:
 
public slots:
    //早期qt版本   必须写到public slots,高版本可以写到public或者全局下
    //返回值 void,需要声明也需要实现
    //可以有参数,可以发生重载
    void treat();
};
 
#endif // STUDENT_H

(2)teacher.h

#ifndef TEACHER_H
#define TEACHER_H
 
#include <QObject>
 
class Teacher : public QObject
{
    Q_OBJECT
public:
    explicit Teacher(QObject *parent = nullptr);
 
signals:
    //自定义信号  写到signals下
    //返回值是void,只需要声明,不需要实现
    //可以有参数,可以重载
    void hungry();
 
 
};
 
#endif // TEACHER_H

(3)widget.h

#ifndef WIDGET_H
#define WIDGET_H
 
#include <QWidget>
#include <teacher.h>
#include <student.h>
 
QT_BEGIN_NAMESPACE
namespace Ui { class Widget; }
QT_END_NAMESPACE
 
class Widget : public QWidget
{
    Q_OBJECT
 
public:
    Widget(QWidget *parent = nullptr);
    ~Widget();
 
private:
    Ui::Widget *ui;
    Teacher * zt;
    Student * st;
 
    void classIsOver();
 
};
#endif // WIDGET_H

(4)student.cpp

#include "student.h"
#include <QDebug.h>
Student::Student(QWidget *parent) : QWidget(parent)
{
 
}
 
void Student::treat()
{
    qDebug()<<"请老师吃饭";
 
}
void Student::treat(QString foodName)
{
 
    //QsString ----》char *先转成OByteArray(.toUtf8())再转char *()
 
    qDebug()<<"请老师吃饭,老师要吃"<<foodName.toUtf8().data();
}

(5)teacher.cpp

#include "teacher.h"
 
Teacher::Teacher(QObject *parent) : QObject(parent)
{
 
}

(6)widget.cpp

#include "widget.h"
#include "ui_widget.h"
#include <QPushButton>
 
//Teacher 类  老师类
//Student 类  学生类
//场景:下课后老师触发一个信号,饿了,学生响应信号  请客吃饭
 
Widget::Widget(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::Widget)
{
    ui->setupUi(this);
    //创建一个老师对象
    this->zt = new Teacher(this);
    //创建一个学生对象
    this->st = new Student(this);
 
//    //老师饿了,学生请客的连接
//    connect(zt,&Teacher::hungry,st,&Student::treat);
 
//    //调用下课函数
//    classIsOver();
 
    //连接带参数的信号和槽
    //指针---》地址
    //函数指针----》函数地址
    void(Teacher:: *teacherSignal)(QString) = &Teacher::hungry;
    void(Student:: *studentSlot)(QString) = &Student::treat;
    connect(zt,teacherSignal,st,studentSlot);
 
    //调用下课函数
    classIsOver();
 
    //点击一个  下课的按钮,再触发下课
    QPushButton * btn = new QPushButton("下课",this);
    //重置窗口大小
    this->resize(600,400);
    //点击按钮触发下课
    //connect(btn,&QPushButton::clicked,this,&Widget::classIsOver);
 
    //无参信号和槽的连接
    void(Teacher:: *teacherSignal2)(void) = &Teacher::hungry;
    void(Student:: *studentSlot2)(void) = &Student::treat;
    connect(zt,teacherSignal2,st,studentSlot2);
 
    //信号连接信号
    connect(btn,&QPushButton::clicked,zt,teacherSignal2);
 
    //断开信号
    //disconnect(zt,teacherSignal2,st,studentSlot2);
    
    
    
}
 
void Widget::classIsOver()
{
    //下课函数调用后触发老师饿了的信号
    //emit zt->hungry();
    emit zt->hungry("鱼香肉丝");
 
}
Widget::~Widget()
{
    delete ui;
}
 

4.Lambda表达式

C++11中的Lambda 表达式用于定义并创建匿名的函效对象,以简化编程工作。

[]  标识符   匿名函数

=  值传递

&  引用传递

()  参数     {}  实现体

mutable   修饰  值传递变量,可以修改拷贝出的数据,改变不了本体

返回值  []()  ->  int {}

最常用   [=](){}

    //利用Lambda表达式,实现点击按钮关闭窗口
    QPushButton *btn2 = new QPushButton;
    btn2->setText("关闭");
    btn2->move(100,0);
    btn2->setParent(this);
    connect(btn2,&QPushButton::clicked,this,[=](){
//       this->close();
//       emit zt->hungry("鱼香肉丝");
       btn2->setText("AAAAA");
    });

5.整体框架


目录
相关文章
|
18天前
|
程序员 C++
【Qt】信号与槽(下)
【Qt】信号与槽(下)
|
18天前
|
Linux C++
【Qt】信号与槽(上)
【Qt】信号与槽(上)
【Qt】信号与槽(上)
|
15天前
【qt】有点意思的信号与槽
【qt】有点意思的信号与槽
8 0
|
16天前
【qt】QTcpSocket相关的信号
【qt】QTcpSocket相关的信号
7 0
|
2月前
|
调度
【浅入浅出】Qt多线程机制解析:提升程序响应性与并发处理能力
在学习QT线程的时候我们首先要知道的是QT的主线程,也叫GUI线程,意如其名,也就是我们程序的最主要的一个线程,主要负责初始化界面并监听事件循环,并根据事件处理做出界面上的反馈。但是当我们只限于在一个主线程上书写逻辑时碰到了需要一直等待的事件该怎么办?它的加载必定会带着主界面的卡顿,这时候我们就要去使用多线程。
|
3月前
|
C++
Qt信号和槽
Qt信号和槽
21 2
|
2月前
|
C++
Qt中的信号与槽如何学习?(包括自定义信号)这篇文章告诉你
以现实中的事件来举例的话,例如有两把不同颜色的信号枪,分别是红色,绿色,打响不通颜色的信号枪会触发不同的槽发生,比如说打响红色这个人就跑步,绿色就走步,但是还有一个很重要的机制,那就是连接,我们需要把信号枪去跟这个人的动作连接起来。 如果上面理解没问题的话我们可以把信号和槽看成两个工具,我们最重要的是如何去把这两个工具连接起来。 它的作用可以让我们更加灵活的去使用不同窗口间的切换以及某些事件的连接。
|
2月前
|
数据安全/隐私保护 C++ 计算机视觉
Qt(C++)开发一款图片防盗用水印制作小工具
文本水印是一种常用的防盗用手段,可以将文本信息嵌入到图片、视频等文件中,用于识别和证明文件的版权归属。在数字化和网络化的时代,大量的原创作品容易被不法分子盗用或侵犯版权,因此加入文本水印成为了保护原创作品和维护知识产权的必要手段。 通常情况下,文本水印可以包含版权声明、制作者姓名、日期、网址等信息,以帮助识别文件的来源和版权归属。同时,为了增强防盗用效果,文本水印通常会采用字体、颜色、角度等多种组合方式,使得水印难以被删除或篡改,有效地降低了盗用意愿和风险。 开发人员可以使用图像处理技术和编程语言实现文本水印的功能,例如使用Qt的QPainter类进行文本绘制操作,将文本信息嵌入到图片中,
116 1
Qt(C++)开发一款图片防盗用水印制作小工具
|
16天前
|
监控 C++ 容器
【qt】MDI多文档界面开发
【qt】MDI多文档界面开发
30 0