Qt中Ui名字空间以及setupUi函数的原理和实现

2024-01-05 09:18

本文主要是介绍Qt中Ui名字空间以及setupUi函数的原理和实现,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!


用最新的QtCreator选择GUI的应用会产生含有如下文件的工程


 

下面就简单分析下各部分的功能。

 

.pro文件是供qmake使用的文件,不是本文的重点【不过其实也很简单的】,在此不多赘述。

所以呢,还是从main开始,

[cpp]  view plain copy
  1. #include <QtGui/QApplication>  
  2. #include "mainwindow.h"  
  3. int main(int argc, char *argv[])  
  4. {  
  5. QApplication a(argc, argv);  
  6. MainWindow w;  
  7. w.show();  
  8. return a.exec();  
  9. }  


 

很简单的样子

QApplication a(argc, argv)和a.exec()可以理解为载入了Qt的架构,跑Qt的程序都要有此部,就不多说了。

其中调用了个MainWindow并把它show了出来,具体分析下

下面是mainwindow.h中的内容

[cpp]  view plain copy
  1. #ifndef MAINWINDOW_H  
  2. #define MAINWINDOW_H  
  3. #include <QtGui/QMainWindow>  
  4. namespace Ui  
  5. {  
  6. class MainWindow;  
  7. }  
  8. class MainWindow : public QMainWindow  
  9. {  
  10. Q_OBJECT  
  11. public:  
  12. MainWindow(QWidget *parent = 0);  
  13. ~MainWindow();  
  14. private:  
  15. Ui::MainWindow *ui;  
  16. };  
  17. #endif // MAINWINDOW_H  



 

开始的namespace Ui可能让人有点摸不着头脑,这是因为qt把ui相关的类单独独立了出来,但类名相同,禁用namespace区别【但是就目前的使用来说,感觉这样做不怎么好,后面我会解释原因】

 

声明namespace Ui是因为要调用Ui中的MainWindow,此MainWindow非彼MainWindow,后面涉及的*ui指针会调用它!

 

关于Q_OBJECT就不说了,Qt中与signal和slot相关的类都要这么声明下。

 

仔细看出了构造,析构就没啥了,只有那么个*ui!不过现在如果运行下,也只会生成个窗体而已。

 

下面来看构造函数和析构函数,其实也就是mainwindow.c

[cpp]  view plain copy
  1. #include "mainwindow.h"  
  2. #include "ui_mainwindow.h"  
  3. MainWindow::MainWindow(QWidget *parent)  
  4. : QMainWindow(parent), ui(new Ui::MainWindow)  
  5. {  
  6. ui->setupUi(this);  
  7. }  
  8. MainWindow::~MainWindow()  
  9. {  
  10. delete ui;  
  11. }  

 

构造时在堆上new了个Ui域中的MainWindow,并调用setupUi,析构仅仅是将其delete了,还是很简单!

 

正如前面所述Qt很好的把ui分离了出去,前面图中的那个.ui文件就是让QtDesigner使的布局用文件!

 

现在运行下,会生成ui_mainwindow.h,这个里面会涉及到真正布局用的函数,也就是那个Ui域中的MainWindow.下面具体看一下,

[cpp]  view plain copy
  1. #ifndef UI_MAINWINDOW_H  
  2. #define UI_MAINWINDOW_H  
  3. #include <QtCore/QVariant>  
  4. #include <QtGui/QAction>  
  5. #include <QtGui/QApplication>  
  6. #include <QtGui/QButtonGroup>  
  7. #include <QtGui/QHeaderView>  
  8. #include <QtGui/QMainWindow>  
  9. #include <QtGui/QMenuBar>  
  10. #include <QtGui/QStatusBar>  
  11. #include <QtGui/QToolBar>  
  12. #include <QtGui/QWidget>  
  13. QT_BEGIN_NAMESPACE  
  14. class Ui_MainWindow  
  15. {  
  16. public:  
  17. QMenuBar *menuBar;  
  18. QToolBar *mainToolBar;  
  19. QWidget *centralWidget;  
  20. QStatusBar *statusBar;  
  21. void setupUi(QMainWindow *MainWindow)  
  22. {  
  23. if (MainWindow->objectName().isEmpty())  
  24. MainWindow->setObjectName(QString::fromUtf8("MainWindow"));  
  25. MainWindow->resize(600, 400);  
  26. menuBar = new QMenuBar(MainWindow);  
  27. menuBar->setObjectName(QString::fromUtf8("menuBar"));  
  28. MainWindow->setMenuBar(menuBar);  
  29. mainToolBar = new QToolBar(MainWindow);  
  30. mainToolBar->setObjectName(QString::fromUtf8("mainToolBar"));  
  31. MainWindow->addToolBar(mainToolBar);  
  32. centralWidget = new QWidget(MainWindow);  
  33. centralWidget->setObjectName(QString::fromUtf8("centralWidget"));  
  34. MainWindow->setCentralWidget(centralWidget);  
  35. statusBar = new QStatusBar(MainWindow);  
  36. statusBar->setObjectName(QString::fromUtf8("statusBar"));  
  37. MainWindow->setStatusBar(statusBar);  
  38. retranslateUi(MainWindow);  
  39. QMetaObject::connectSlotsByName(MainWindow);  
  40. // setupUi  
  41. void retranslateUi(QMainWindow *MainWindow)  
  42. {  
  43. MainWindow->setWindowTitle(QApplication::translate("MainWindow""MainWindow", 0, QApplication::UnicodeUTF8));  
  44. Q_UNUSED(MainWindow);  
  45. // retranslateUi  
  46. };  
  47. namespace Ui {  
  48. class MainWindow: public Ui_MainWindow {};  
  49. // namespace Ui  
  50. QT_END_NAMESPACE  
  51. #endif // UI_MAINWINDOW_H  




 

吼吼,一下子多了不少,但其实还是很容易的。Ui_MainWindow声明了几个构件,具体我就不说了,因为也没啥可说的,它实现了setupUi函式,也就是前面那个MainWindow中调用的setupUi。

但是要说明的是QMetaObject::connectSlotsByName函式会自动连接相应名称的信号与槽,但要注意它连接的是传入的MainWindow及其子构件【不是子类】,注意前边ui->setupUi(this)中传入的this,也就是非ui域中的MainWindow,所以如果要声明signalslot时还是要在非ui域的MainWindow中来声明,然后通过ui->xxx的形式来与GUI产生交互!如果我们在QtDesiner中拖放一个按钮然后点击go to slot就很容易印证这一点。

retranslateUi则会为ui中的构件命名,具体也不在此多说。

最后还是看看这段代码

namespace Ui {

class MainWindow: public Ui_MainWindow {};

} // namespace Ui

前面非Ui域中的MainWindow的*ui指向的是Ui域中的MainWindow,而Ui域中的MainWindow出了继承了Ui_MainWindow之外,内部一贫如洗!【有点绕口了】

来张图片,再复习下

 

最后要说明的有两点,个人感觉是QtCreator的BUG,

其一是如果自己定制控件,并且想在内置的designer中载入,win下用mingw是不可行的,因为sdk套件中的designer是用微软的编译器编译的,当然也有个比较方便的解决的办法,就是把qtcreator的源码下来,用现有的creator再编译一遍,然后覆盖过去就行了。

其二也是前面提到的,两个同名的MainWindow仅用Ui域来区分,虽然感觉这样做从设计上来说是很美的,但调试时却会有些许的问题,总之在creator中调试不能识别正确的域,具体见下图例


 

像上面这张图this实际上应该指向的是Ui域中的MainWindow【this其实指向的是MainWindow,它并不知是哪个域的MainWindow,再往下展开就错误的指向了Ui域】,但调试的数据区指向了Ui域中的MainWindow,当然也不是没有解决的办法,你可以手工将Ui域中的MainWindow改下名就可以获得正确的调试信息了,只是这样做稍显麻烦,而且再度运行qmake后可能还要重新修改。

这篇关于Qt中Ui名字空间以及setupUi函数的原理和实现的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/572382

相关文章

Django中的函数视图和类视图以及路由的定义方式

《Django中的函数视图和类视图以及路由的定义方式》Django视图分函数视图和类视图,前者用函数处理请求,后者继承View类定义方法,路由使用path()、re_path()或url(),通过in... 目录函数视图类视图路由总路由函数视图的路由类视图定义路由总结Django允许接收的请求方法http

python使用Akshare与Streamlit实现股票估值分析教程(图文代码)

《python使用Akshare与Streamlit实现股票估值分析教程(图文代码)》入职测试中的一道题,要求:从Akshare下载某一个股票近十年的财务报表包括,资产负债表,利润表,现金流量表,保存... 目录一、前言二、核心知识点梳理1、Akshare数据获取2、Pandas数据处理3、Matplotl

分布式锁在Spring Boot应用中的实现过程

《分布式锁在SpringBoot应用中的实现过程》文章介绍在SpringBoot中通过自定义Lock注解、LockAspect切面和RedisLockUtils工具类实现分布式锁,确保多实例并发操作... 目录Lock注解LockASPect切面RedisLockUtils工具类总结在现代微服务架构中,分布

Java使用Thumbnailator库实现图片处理与压缩功能

《Java使用Thumbnailator库实现图片处理与压缩功能》Thumbnailator是高性能Java图像处理库,支持缩放、旋转、水印添加、裁剪及格式转换,提供易用API和性能优化,适合Web应... 目录1. 图片处理库Thumbnailator介绍2. 基本和指定大小图片缩放功能2.1 图片缩放的

Python使用Tenacity一行代码实现自动重试详解

《Python使用Tenacity一行代码实现自动重试详解》tenacity是一个专为Python设计的通用重试库,它的核心理念就是用简单、清晰的方式,为任何可能失败的操作添加重试能力,下面我们就来看... 目录一切始于一个简单的 API 调用Tenacity 入门:一行代码实现优雅重试精细控制:让重试按我

MySQL常用字符串函数示例和场景介绍

《MySQL常用字符串函数示例和场景介绍》MySQL提供了丰富的字符串函数帮助我们高效地对字符串进行处理、转换和分析,本文我将全面且深入地介绍MySQL常用的字符串函数,并结合具体示例和场景,帮你熟练... 目录一、字符串函数概述1.1 字符串函数的作用1.2 字符串函数分类二、字符串长度与统计函数2.1

Redis客户端连接机制的实现方案

《Redis客户端连接机制的实现方案》本文主要介绍了Redis客户端连接机制的实现方案,包括事件驱动模型、非阻塞I/O处理、连接池应用及配置优化,具有一定的参考价值,感兴趣的可以了解一下... 目录1. Redis连接模型概述2. 连接建立过程详解2.1 连php接初始化流程2.2 关键配置参数3. 最大连

Python实现网格交易策略的过程

《Python实现网格交易策略的过程》本文讲解Python网格交易策略,利用ccxt获取加密货币数据及backtrader回测,通过设定网格节点,低买高卖获利,适合震荡行情,下面跟我一起看看我们的第一... 网格交易是一种经典的量化交易策略,其核心思想是在价格上下预设多个“网格”,当价格触发特定网格时执行买

python设置环境变量路径实现过程

《python设置环境变量路径实现过程》本文介绍设置Python路径的多种方法:临时设置(Windows用`set`,Linux/macOS用`export`)、永久设置(系统属性或shell配置文件... 目录设置python路径的方法临时设置环境变量(适用于当前会话)永久设置环境变量(Windows系统

python使用try函数详解

《python使用try函数详解》Pythontry语句用于异常处理,支持捕获特定/多种异常、else/final子句确保资源释放,结合with语句自动清理,可自定义异常及嵌套结构,灵活应对错误场景... 目录try 函数的基本语法捕获特定异常捕获多个异常使用 else 子句使用 finally 子句捕获所