• QT基础学习


    2创建项目

    2.1使用向导创建

    打开Qt Creator 界面选择 New Project或者选择菜单栏 【文件】-【新建文件或项目】菜单项

    弹出New Project对话框,选择Qt Widgets Application,

    选择【Choose】按钮,弹出如下对话框

    设置项目名称和路径,按照向导进行下一步,(不能有中文和空格

    选择编译套件

    向导会默认添加一个继承自CMainWindow的类,可以在此修改类的名字和基类。默认的基类有QMainWindow、QWidget以及QDialog三个,我们可以选择QWidget(类似于空窗口),这里我们可以先创建一个不带UI的界面,继续下一步

    系统会默认给我们添加main.cpp、mywidget.cpp、 mywidget.h和一个.pro项目文件,点击完成,即可创建出一个Qt桌面程序。

    对main原始代码的一些解释

    1. #include "widget.h"
    2. #include //包含一个应用程序类的头文件
    3. //main程序入口 argc命令行变量的数量 argv命令行变量的数组
    4. int main(int argc, char *argv[])
    5. {
    6. //a应用程序对象,在gt中,应用程序对象 有且仅有一个
    7. QApplication a(argc, argv);
    8. //窗口对象 mywidget父类 ->Qwidget
    9. Widget w;
    10. //窗口对象 默认不会显示,必须要调用show方法显示窗口
    11. w.show();
    12. //让应用程序对象进入消息循环//当代码阻塞到这行
    13. return a.exec();
    14. // while (true)
    15. // {
    16. // if(点击叉子)
    17. // {
    18. // break;
    19. // }
    20. // }
    21. }
    1. QT += core gui //Qt包含的模块
    2. greaterThan(QT_MAJOR_VERSION, 4): QT += widgets //大于4版本以上 包含 widget模块
    3. TARGET = QTtest //目标 生成的exe程序的名称
    4. TEMPLATE = app
    5. DEFINES += QT_DEPRECATED_WARNINGS
    6. before Qt 6.0.0
    7. SOURCES += \ //源文件
    8. main.cpp \
    9. widget.cpp
    10. HEADERS += \ //头文件
    11. widget.h
    总结

    快捷键   

    帮助文档 第一种方式 F1 第二种 左侧按钮 第三种 D:\QT\5.11.1\mingw53_32\bin

    1. // 命名规范
    2. // 类名 首字母大写,单词和单词之间首字母大写
    3. // 函数名 变量名称 首字母小写,单词和单词之间首字母大写
    4. //快捷键//注释 ctrl+/
    5. //运行 etrl + r//编译 ctrl + b
    6. //字体缩放 ctrl + 鼠标滚轮
    7. //查找 ctrl + f
    8. //整行移动 ctrl + shift+,或者↓
    9. //帮助文档 F1
    10. //自动对齐 ctrl + i;
    11. //同名之间的.h 和.cpp切换 F4

    3第一个Qt小程序

    3.1 按钮的创建

    mywidget.h

    1. #ifndef MYWIDGET_H
    2. #define MYWIDGET_H
    3. #include
    4. class myWidget : public QWidget //继承的语法
    5. {
    6. Q_OBJECT
    7. public:
    8. myWidget(QWidget *parent = 0); //有参构造函数
    9. ~myWidget();
    10. };
    11. #endif // MYWIDGET_H

     mypushbutton.h

    1. #ifndef MYPUSHBUTTON_H
    2. #define MYPUSHBUTTON_H
    3. #include
    4. class MyPushButton : public QPushButton
    5. {
    6. Q_OBJECT
    7. public:
    8. explicit MyPushButton(QWidget *parent = 0);
    9. ~MyPushButton();
    10. signals:
    11. public slots:
    12. };
    13. #endif // MYPUSHBUTTON_H

     main.cpp

    1. #include "mywidget.h"
    2. #include
    3. int main(int argc, char *argv[])
    4. {
    5. QApplication a(argc, argv);
    6. myWidget w; //窗口对象 myWidget的父类 -> QWidget
    7. w.show(); //窗口对象 默认不会显示,必须要调用show方法显示窗口
    8. return a.exec();//让应用程序对象进入消息循环//当代码阻塞到这行
    9. // while(true)
    10. // {
    11. // if(点击叉子)
    12. // {
    13. // break;
    14. // }
    15. // }
    16. }

     mypushbutton.cpp

    1. #include "widget.h"
    2. Widget::Widget(QWidget *parent)
    3. : QWidget(parent)
    4. {
    5. //创建一个按钮
    6. QPushButton * btn = new QPushButton;
    7. //btn->show(); //show以顶层方式弹出窗口控件
    8. //让btn对象依赖在mywidget窗口中
    9. btn->setParent (this);
    10. //显示文本
    11. btn->setText ("第一个按钮");
    12. //创建第二个按钮 按照控件的大小创建窗口
    13. QPushButton * btn2 = new QPushButton("第二个按钮"this):
    14. //移动btn2按钮
    15. btn2->move (100,100);
    16. //重置窗口大小
    17. resize (600, 400);
    18. //设置固定窗口大小
    19. setFixedSize (600, 400);
    20. //设置窗口标题
    21. setwindowTitle("第一个窗口");
    22. Widget::~Widget()
    23. {
    24. }

     mywidget.h

    1. #include "mywidget.h"
    2. #include //按钮控件的头文件
    3. #include "mypushbutton.h"
    4. #include
    5. myWidget::myWidget(QWidget *parent)
    6. : QWidget(parent)//初始化列表
    7. {
    8. //创建一个按钮
    9. QPushButton * btn = new QPushButton;
    10. //btn->show(); //show以顶层方式弹出窗口控件
    11. //让btn对象 依赖在 myWidget窗口中
    12. btn->setParent(this);
    13. //显示文本
    14. btn->setText("第一个按钮");
    15. //创建第二个按钮 按照控件的大小创建窗口
    16. QPushButton * btn2 = new QPushButton("第二个按钮",this);
    17. //移动btn2按钮
    18. btn2->move(100,100);
    19. //按钮可不可以 重新制定大小 可以!
    20. btn2->resize(50,50);
    21. //重置窗口大小
    22. resize(600,400);
    23. //设置固定窗口大小
    24. setFixedSize(600,400);
    25. //设置窗口标题
    26. setWindowTitle("第一个窗口");
    27. //创建一个自己的按钮对象
    28. MyPushButton * myBtn = new MyPushButton;
    29. myBtn->setText("我自己的按钮");
    30. myBtn->move(200,0);
    31. myBtn->setParent(this); //设置到对象树中
    32. //需求 点击我的按钮 关闭窗口
    33. //参数1 信号的发送者 参数2发送的信号(函数的地址)参数3 信号的接受者 参数4 处理的槽函数
    34. connect( myBtn, &MyPushButton::clicked, this, &myWidget::close );
    35. connect( myBtn, &QPushButton::clicked, this, &QWidget::close );
    36. }
    37. myWidget::~myWidget()
    38. {
    39. }

    来自ChatGPT的代码解析解答

    总结

    3.2对象模型(对象树)

    在Qt中创建对象的时候会提供一个Parent对象指针,下面来解释这个parent到底是干什么的。

     

    • QObject是以对象树的形式组织起来的。
    1. 当你创建一个QObject对象时,会看到QObject的构造函数接收一个QObject指针作为参数,这个参数就是 parent,也就是父对象指针。
    2. 这相当于,在创建QObject对象时,可以提供一个其父对象,我们创建的这个QObject对象会自动添加到其父对象的children()列表。
    3. 当父对象析构的时候,这个列表中的所有对象也会被析构。(注意,这里的父对象并不是继承意义上的父类!
    4. 这种机制在 GUI 程序设计中相当有用。例如,一个按钮有一个QShortcut(快捷键)对象作为其子对象。当我们删除按钮的时候,这个快捷键理应被删除。这是合理的。
    • QWidget是能够在屏幕上显示的一切组件的父类。
    1. QWidget继承自QObject,因此也继承了这种对象树关系。一个孩子自动地成为父组件的一个子组件。因此,它会显示在父组件的坐标系统中,被父组件的边界剪裁。例如,当用户关闭一个对话框的时候,应用程序将其删除,那么,我们希望属于这个对话框的按钮、图标等应该一起被删除。事实就是如此,因为这些都是对话框的子组件。
    2. 当然,我们也可以自己删除子对象,它们会自动从其父对象列表中删除。比如,当我们删除了一个工具栏时,其所在的主窗口会自动将该工具栏从其子对象列表中删除,并且自动调整屏幕显示。

    Qt 引入对象树的概念,在一定程度上解决了内存问题。

    •  当一个QObject对象在堆上创建的时候,Qt 会同时为其创建一个对象树。不过,对象树中对 象的顺序是没有定义的。这意味着,销毁这些对象的顺序也是未定义的。
    • 任何对象树中的 QObject对象 delete 的时候,如果这个对象有 parent,则自动将其从 parent 的children()列表中删除;如果有孩子,则自动 delete 每一个孩子。Qt 保证没有QObject会被 delete 两次,这是由析构顺序决定的。
    • 如果QObject在栈上创建,Qt 保持同样的行为。正常情况下,这也不会发生什么问题。来看下下面的代码片段:

      1. {
      2. QWidget window;
      3. QPushButton quit("Quit", &window);
      4. }

      作为父组件的 window 和作为子组件的 quit 都是QObject的子类(事实上,它们都是QWidget的子类,而QWidget是QObject的子类)。这段代码是正确的,quit 的析构函数不会被调用两次,因为标准 C++要求,局部对象的析构顺序应该按照其创建顺序的相反过程。因此,这段代码在超出作用域时,会先调用 quit 的析构函数,将其从父对象 window 的子对象列表中删除,然后才会再调用 window 的析构函数。

      但是,如果我们使用下面的代码:

      1. {
      2. QPushButton quit("Quit");
      3. QWidget window;
      4. quit.setParent(&window);
      5. }

      情况又有所不同,析构顺序就有了问题。我们看到,在上面的代码中,作为父对象的 window 会首先被析构,因为它是最后一个创建的对象。在析构过程中,它会调用子对象列表中每一个对象的析构函数,也就是说, quit 此时就被析构了。然后,代码继续执行,在 window 析构之后,quit 也会被析构,因为 quit 也是一个局部变量,在超出作用域的时候当然也需要析构。但是,这时候已经是第二次调用 quit 的析构函数了,C++ 不允许调用两次析构函数,因此,程序崩溃了。

      由此我们看到,Qt 的对象树机制虽然帮助我们在一定程度上解决了内存问题,但是也引入了一些值得注意的事情。这些细节在今后的开发过程中很可能时不时跳出来烦扰一下,所以,我们最好从开始就养成良好习惯,在 Qt 中,尽量在构造的时候就指定 parent 对象,并且大胆在堆上创建。

    3.3 Qt窗口坐标体系 

    坐标体系

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

     对于嵌套窗口,其坐标是相对于父窗口来说的。

    4 信号和槽机制

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

    信号:各种事件

    槽: 响应信号的动作

    当某个事件发生后,如某个按钮被点击了一下,它就会发出一个被点击的信(signal)。

    某个对象接收到这个信号之后,就会做一些相关的处理动作(称为槽slot)。

    但是Qt对象不会无故收到某个信号,要想让一个对象收到另一个对象发出的信号,这时候需要建立连接(connect)

    4.1 系统自带的信号和槽

            下面我们完成一个小功能,上面我们已经学习了按钮的创建,但是还没有体现出按钮的功能,按钮最大的功能也就是点击后触发一些事情,比如我们点击按钮,就把当前的窗口给关闭掉,那么在Qt中,这样的功能如何实现呢?

    如下图:

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

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

               其实两行代码就可以搞定了,我们看下面的代码

    1. QPushButton * quitBtn = new QPushButton("关闭窗口",this);
    2. connect(quitBtn,&QPushButton::clicked,this,&MyWidget::close);

        第一行是创建一个关闭按钮,这个之前已经学过,第二行就是核心了,也就是信号槽的使用方式

        connect函数是建立信号发送者、信号、信号接收者、槽四者关系的函数:

    connect(sender, signal, receiver, slot);

    参数解释:

            1)sender:信号发送者

            2)signal:信号

            3)receiver:信号接收者

            4)slot:接收对象在接收到信号之后所需要调用的函数(槽函数)

             这里要注意的是connect的四个参数都是指针,信号和槽是函数指针。

            系统自带的信号和槽如何查找呢,这个就需要利用帮助文档了,在帮助文档中比如我们上面的按钮的点击信号,在帮助文档中输入QPushButton,首先我们可以在Contents中寻找关键字 signals,信号的意思,但是我们发现并没有找到,这时候我们应该想到也许这个信号的被父类继承下来的,因此我们去他的父类QAbstractButton中就可以找到该关键字,点击signals索引到系统自带的信号有如下几个。

            这里的clicked就是我们要找到,槽函数的寻找方式和信号一样,只不过他的关键字是slot。 

    4.2 自定义信号和槽

    使用connect()可以让我们连接系统提供的信号和槽。但是,Qt 的信号槽机制并不仅仅是使用系统提供的那部分,还会允许我们自己设计自己的信号和槽。

    4.2.1自定义信号使用条件
    1. 声明在类的signals域下
    2. 没有返回值,void类型的函数
    3. 只有函数声明,没有定义
    4. 可以有参数,可以重载
    5. 通过emit关键字来触发信号,形式:emit object->sig(参数);
     4.2.3 使用自定义信号和槽

            定义场景:下课了,老师跟同学说肚子饿了(信号),学生请老师吃饭(槽)

            首先定义一个学生类和老师类:

            老师类中声明信号 饿了 hungry

    1. signals:
    2. void hungry();

          学生类中声明槽 请客treat

    1. public slots:
    2. void treat();

        在窗口中声明一个公共方法下课,这个方法的调用会触发老师饿了这个信号,而响应槽函数学生请客

    1. void MyWidget::ClassIsOver()
    2. {
    3. //发送信号
    4. emit teacher->hungry();
    5. }

            学生响应了槽函数,并且打印信息

            //自定义槽函数 实现

    1. void Student::treat()
    2. {
    3. qDebug() << "Student treat teacher";
    4. }

           在窗口中连接信号槽

    1. teacher = new Teacher(this);
    2. student = new Student(this);
    3. connect(teacher,&Teacher::hungury,student,&Student::treat);

            并且调用下课函数,测试打印出相应log 

    4.3 信号和槽的扩展

            1.一个信号可以和多个槽相连

            如果是这种情况,这些槽会一个接一个的被调用,但是槽函数调用顺序是不确定的。像上面的例子,可以将一个按钮点击信号连接到关闭窗口的槽函数,同时也连接到学生请吃饭的槽函数,点击按钮的时候可以看到关闭窗口的同时也学生请吃饭的log也打印出来。

            2. 多个信号可以连接到一个槽

            只要任意一个信号发出,这个槽就会被调用。如:一个窗口多个按钮都可以关闭这个窗口。

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

            当第一个信号发出时,第二个信号被发出。除此之外,这种信号-信号的形式和信号-槽的形式没有什么区别。注意这里还是使用connect函数,只是信号的接收者和槽函数换成另一个信号的发送者和信号函数。如上面老师饿了的例子,可以新建一个按钮btn。

    connect(btn,&QPushButton::clicked,teacher,&Teacher::hungry);

            4.信号和槽可以断开连接

            可以使用disconnect函数,当初建立连接时connect参数怎么填的,disconnect里边4个参数也就怎么填。这种情况并不经常出现,因为当一个对象delete之后,Qt自动取消所有连接到这个对象上面的槽。

            5.信号和槽函数参数类型和个数必须同时满足两个条件

                    1)信号函数的参数个数必须大于等于槽函数的参数个数

                    2)信号函数的参数类型和槽函数的参数类型必须一一对应

     自定义信号槽需要注意的事项

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

    4.4 Qt4版本的信号槽写法 

    1. connect(
    2. teacher,
    3. SIGNAL(hungry(QString)),
    4. student,
    5. SLOT(treat(QString))
    6. );

            这里使用了SIGNAL和SLOT这两个宏,宏的参数是信号函数和槽函数的函数原型。

            因为直接填入了函数原型,所有这里边编译不会出现因为重载导致的函数指针二义性的问题。但问题是如果函数原型填错了,或者不符合信号槽传参个数类型约定,编译期间也不会报错,只有运行期间才会看到错误log输出。

            原因就是这两个宏将后边参数(函数原型)转化成了字符串。目前编译器还没有那么智能去判断字符串里边的内容符不符合运行条件。

     4.5 Lambda 表达式

            C++11中的Lambda表达式用于定义匿名的函数对象,以简化编程工作。首先看一下Lambda表达式的基本构成:

            分为四个部分:[局部变量捕获列表]、(函数参数)、函数额外属性设置opt、函数返回值->retype、{函数主体}

    1. [capture](parameters) opt ->retType
    2. {
    3. ……;
    4. }
    4.5.1 局部变量引入方式 

            [ ],标识一个Lambda的开始。由于lambda表达式可以定义在某一个函数体A里边,所以lambda表达式有可能会去访问A函数中的局部变量。中括号里边内容是描述了在lambda表达式里边可以使用的外部局部变量的列表:

            []

                表示lambda表达式不能访问外部函数体的任何局部变量

            [a]

                在函数体内部使用值传递的方式访问a变量

            [&b]

                在函数体内部使用引用传递的方式访问b变量

            [=]

                函数外的所有局部变量都通过值传递的方式使用, 函数体内使用的是副本

            [&]

                引用的方式使用lambda表达式外部的所有变量

            [=, &foo]

                foo使用引用方式, 其余是值传递的方式

            [&,foo]

                foo使用值传递方式, 其余是引用传递的方式

             [this]

               在函数内部可以使用类的成员函数和成员变量,=和&形式也都会默认引入

            由于引用方式捕获对象会有局部变量释放了而lambda函数还没有被调用的情况。如果执行lambda函数那么引用传递方式捕获进来的局部变量的值不可预知。

            所以在无特殊情况下建议使用[=](){}的形式

     4.5.2 函数参数

            (params)表示lambda函数对象接收的参数,类似于函数定义中的小括号表示函数接收的参数类型和个数。参数可以通过按值(如:(int a,int b))和按引用(如:(int &a,int &b))两种方式进行传递。函数参数部分可以省略,省略后相当于无参的函数。

    4.5.3 选项 Opt

            Opt 部分是可选项,最常用的是mutable声明,这部分可以省略。外部函数局部变量通过值传递引进来时,其默认是const,所以不能修改这个局部变量的拷贝,加上mutable就可以

    1. int a = 10 ;
    2. [=]()
    3. {
    4. a=20;//编译报错,a引进来是const
    5. }
    6. [=]()mutable
    7. {
    8. a=20;//编译成功
    9. };
    4.5.5 是函数主体{}

            {},标识函数的实现,这部分不能省略,但函数体可以为空。

     4.5.6 槽函数使用 Lambda 表达式

    以QPushButton点击事件为例:

    1. connect(btn,&QPushButton::clicked,[=](){
    2. qDebug()<<"Clicked";
    3. });

            这里可以看出使用Lambda表达式作为槽的时候不需要填入信号的接收者。当点击按钮的时候,clicked信号被触发,lambda表达式也会直接运行。当然lambda表达式还可以指定函数参数,这样也就能够接收到信号函数传递过来的参数了。

            由于lambda表达式比我们自己自定义槽函数要方便而且灵活得多,所以在实现槽函数的时候优先考虑使用Lambda表达式。一般我们的使用习惯也是lambda表达式外部函数的局部变量全部通过值传递捕获进来,也就是:

    [=](){  }的形式

     老师学生吃饭例子:

    student.h

    teacher.h
    widget.h

    main.cpp
    student.cpp

    teacher.cpp

    widget.cpp
    1. #include "widget.h"
    2. #include "ui_widget.h"
    3. #include <QPushButton>
    4. #include <QDebug>
    5. //Teacher 类 老师类
    6. //Student 类 学生类
    7. //下课后 ,老师会触发一个信号,饿了 ,学生响应信号,请客吃饭
    8. void func()
    9. {
    10. qDebug() <<"aaa";
    11. }
    12. Widget::Widget(QWidget *parent) :
    13. QWidget(parent),
    14. ui(new Ui::Widget)
    15. {
    16. ui->setupUi(this);
    17. //创建一个老师对象
    18. this->zt = new Teacher(this);
    19. //创建一个学生对象
    20. this->st = new Student(this);
    21. // //老师饿了 学生请客的连接
    22. // connect(zt,&Teacher::hungry,st,&Student::treat);
    23. // //调用下课函数
    24. // classIsOver();
    25. //连接带参数的 信号和槽
    26. //指针 -> 地址
    27. // 函数指针 -> 函数地址
    28. void(Teacher:: *teacherSignal)(QString ) = &Teacher::hungry;
    29. void(Student:: *studentSlot)(QString ) = &Student::treat;
    30. connect(zt,teacherSignal,st,studentSlot);
    31. // classIsOver();
    32. //点击一个 下课的按钮 ,再触发下课
    33. QPushButton * btn = new QPushButton("下课",this);
    34. //重置窗口大小
    35. this->resize(600,400);
    36. //点击按钮 触发下课
    37. //connect(btn,&QPushButton::clicked,this,&Widget::classIsOver);
    38. //无参信号和槽连接
    39. void(Teacher:: *teacherSignal2)(void) = &Teacher::hungry;
    40. void(Student:: *studentSlot2)(void) = &Student::treat;
    41. //connect(zt,teacherSignal2,st,studentSlot2);
    42. //信号连接信号
    43. connect(btn,&QPushButton::clicked, zt, teacherSignal2);
    44. //断开信号
    45. //disconnect(zt,teacherSignal2,st,studentSlot2);
    46. //拓展
    47. //1、信号是可以连接信号
    48. //2、一个信号可以连接多个槽函数
    49. //3、多个信号 可以连接 同一个槽函数
    50. //4、信号和槽函数的参数 必须类型一一对应
    51. //5、信号和槽的参数个数 是不是要一致?信号的参数个数 可以多余槽函数的参数个数
    52. //Qt4版本以前的信号和槽连接方式
    53. //利用Qt4信号槽 连接无参版本
    54. //Qt4版本 底层SIGNAL("hungry") SLOT( "treat")
    55. connect(zt,SIGNAL(hungry()) , st , SLOT(treat()));
    56. //Qt4版本优点:参数直观,缺点 :类型不做检测
    57. //Qt5以上 支持 Qt4的版本写法,反之不支持
    58. // QPushButton * btn2 = new QPushButton;
    59. // [btn](){
    60. // btn->setText("aaaa");
    61. // btn2->setText("bbb"); //btn2看不到
    62. // }();
    63. // mutable关键字 用于修饰值传递的变量,修改的是拷贝,而不是本体
    64. // QPushButton * myBtn = new QPushButton (this);
    65. // QPushButton * myBtn2 = new QPushButton (this);
    66. // myBtn2->move(100,100);
    67. // int m = 10;
    68. // connect(myBtn,&QPushButton::clicked,this,[m] ()mutable { m = 100 + 10; qDebug() << m; });
    69. // connect(myBtn2,&QPushButton::clicked,this,[=] () { qDebug() << m; });
    70. // qDebug() << m;
    71. // int ret = []()->int{return 1000;}();
    72. // qDebug() << "ret = " << ret ;
    73. //利用lambda表达式 实现点击按钮 关闭窗口
    74. QPushButton * btn2 = new QPushButton ;
    75. btn2->setText("关闭");
    76. btn2->move(100,0);
    77. btn2->setParent(this);
    78. connect(btn2,&QPushButton::clicked, [=](){
    79. // this->close();
    80. // emit zt->hungry("宫保鸡丁");
    81. btn2->setText("aaaa");
    82. });
    83. //lambda表达式 最常用 [=](){}
    84. }
    85. void Widget::classIsOver()
    86. {
    87. //下课函数,调用后 触发老师饿了的信号
    88. //emit zt->hungry();
    89. emit zt->hungry("宫保鸡丁");
    90. }
    91. Widget::~Widget()
    92. {
    93. delete ui;
    94. }

     

    5QMainWindow

            QMainWindow是一个为用户提供主窗口程序的类,包含一个菜单栏(menu bar)、多个工具栏(tool bars)、多个锚接部件(dock widgets)、一个状态栏(status bar)及一个中心部件(central widget),是许多应用程序的基础,如文本编辑器,图片编辑器等。

    5.1 菜单栏

            一个主窗口最多只有一个菜单栏。位于主窗口顶部、主窗口标题栏下面。

           1.通过QMainWindow类的menubar()函数获取主窗口菜单栏指针,如果当前窗口没有菜单栏,该函数会自动创建一个。

       QMenuBar *  menuBar() const;

            2.创建菜单,调用QMenu的成员函数addMenu来添加菜单

    1. QAction* addMenu(QMenu * menu);
    2. QMenu* addMenu(const QString & title);
    3. QMenu* addMenu(const QIcon & icon, const QString & title);

            3.创建菜单项,调用QMenu的成员函数addAction来添加菜单项

    1. QAction* activeAction() const;
    2. QAction* addAction(const QString & text);
    3. QAction* addAction(const QIcon & icon, const QString & text);
    4. QAction* addAction(const QString & text, const QObject * receiver,
    5. const char * member, const QKeySequence & shortcut = 0);
    6. QAction* addAction(const QIcon & icon, const QString & text,
    7. const QObject * receiver, const char * member,
    8. const QKeySequence & shortcut = 0);

             Qt 并没有专门的菜单项类,只是使用一个QAction类,抽象出公共的动作。当我们把QAction对象添加到菜单,就显示成一个菜单项,添加到工具栏,就显示成一个工具按钮。用户可以通过点击菜单项、点击工具栏按钮、点击快捷键来激活这个动作。

     5.2 工具栏

            主窗口的工具栏上可以有多个工具条,通常采用一个菜单对应一个工具条的的方式,也可根据需要进行工具条的划分。

            1.调用QMainWindowd对象的成员函数addToolBar(),该函数每次调用都会创建一个新的工具栏,并且返回该工具栏的指针。

            2.插入属于工具条的项,这时工具条上添加项也是用QAction。通过QToolBar类的addAction函数添加。

            1)Qt::LeftToolBarArea           停靠在左侧

            2)Qt::RightToolBarArea         停靠在右侧

            3)Qt::TopToolBarArea           停靠在顶部

            4)Qt::BottomToolBarArea   停靠在底部       

            5)Qt::AllToolBarAreas           以上四个位置都可停靠

    使用setAllowedAreas()函数指定停靠区域:

    setAllowedAreas(Qt::LeftToolBarArea | Qt::RightToolBarArea

    使用setFloatable(trueOrFalse)函数来设定工具栏可否浮动

    使用setMoveable(trueOrFalse)函数设定工具栏的可移动性:

    setMoveable(false)//工具条不可移动, 只能停靠在初始化的位置上

     5.3 状态栏

            一个QMainWindow的程序最多只有一个状态栏。QMainWindow中可以有多个的部件都使用add…名字的函数,而只有一个的部件,就直接使用获取部件的函数,如menuBar。同理状态栏也提供了一个获取状态栏的函数statusBar(),没有就自动创建一个并返回状态栏的指针。
     

     QMenuBar *	menuBar() const;

    1.添加小部件(从状态栏左侧添加)

    1. void addWidget(QWidget * widget, int stretch = 0);
    2. //插入小部件
    3. int insertWidget(int index, QWidget * widget, int stretch = 0);
    4. //删除小部件
    5. void removeWidget(QWidget * widget);

    2.添加小部件(从状态栏右侧添加)

    void addPermenentWidget (QWidget *widget, int stretch = 0);

    5.4 停靠部件(也成为铆接部件、浮动窗口)

            停靠部件 QDockWidget,也称浮动窗口,可以有多个。

    1. QDockWidget * dock = new QDockWidget("标题",this);
    2. //添加停靠部件到mainWindow中,并且设定默认停靠在左边
    3. addDockWidget(Qt::LeftDockWidgetArea,dock);
    4. //设定停靠部件允许停靠的范围
    5. dock->setAllowedAreas(Qt::LeftDockWidgetArea |
    6. Qt::RightDockWidgetArea | Qt::TopDockWidgetArea);

  • 相关阅读:
    基于遗传算法的车辆和无人机协同路径规划问题研究附Matlab代码
    什么是jnlp
    Doris学习笔记之监控
    字节、字、双字 关系
    公众号网课答案查题方法搭建
    竞赛选题 疲劳驾驶检测系统 python
    SNMP 协议解析(一)
    linux进程管理
    【计算机网络】第三章课后习题答案
    重试机制思考与实现
  • 原文地址:https://blog.csdn.net/LDBH66/article/details/133833496