首页 > 编程语言 >谈谈 Qt 程序安装包的大小,以及简要打包指南

谈谈 Qt 程序安装包的大小,以及简要打包指南

时间:2024-09-07 22:46:29浏览次数:15  
标签:简要 Qt OpenGL 程序安装 程序 exe QML ICU

https://www.jianshu.com/p/0dd884a43de6

 

本文是水木社区KDE与Qt编程技术版版主的文章,我觉得写的很好,就转载过来了,原文地址:http://hgoldfish.com/blogs/article/103/

经常看到网上有些论调说 Qt 程序无比庞大,甚至拿 .NET 程序来比,说 Qt 程序打包以后跟 .NET 安装包差不多大。由此影响了很多人对 Qt 的选择。我觉得有必要对此做一些澄清——

显然这个说法是错误的!!

很容易理解,虽然 Qt 提供了很多组件,但并非所有的组件都会被程序使用,也并非所有的组件都需要打包到程序安装包里面。以 Qt 5.7 为例,一个可以正常使用的 helloworld.exe 程序未压缩不过 20M,使用 lzma 压缩算法,压缩率 25%,压缩完才 6M!

那盛传的 Qt 上百 M 的容量又是怎么回事呢?

这事从头说起。

Qt4.4 引入了 Webkit 方便程序员往自己的程序里面嵌入 HTML 页面。刚开始倒也没事,反正不主动包含到项目里面的话,程序安装包并不会变大。那会儿,Qt 程序只要 8M(压缩到3M)就搞定了。到了 Qt5 的时候 Webkit 换成了 Chromium 内核 Blink,名字改成了 QtWebEngine. Qt 开发者发现 Blink 使用的 ICU,提供了比旧版本 Qt 更完整的国际化能力。于是,据说是因为一个误会,Qt 开发者就在 Qt 5.0 里面强制依赖于 ICU 了。这个 ICU 包含了大量的国际化元信息,一下子增长了近 30M 的容量。。

导致 Qt 容易增长还有另外一个重要原因,也就是 Qt 自从 4.7 以后引入的 QML。这是一个基于 ECMAScript 改造的语言。从此以后,Qt 开发就有两种流派,一者使用原来的 C++ 语言进行开发,另外一种使用 QML 语言进行开发。两者共享一样的底层设施,比如各种数据结构与事件循环 QtCore 和 带有 OpenGL 支持的 QtGui。但在上层,Qt 为 C++ 开发者们提供了 QtWidgets,又为 QML 开发者提供了独立的 QtQuick.

QtQuick 本身只有简单的布局语言,Qt 又在这个基础上提供了 QtControls 模拟 Android,iOS 和 Windows 程序的外观。早期的 Qt5 没有设计好,导致使用 QML 时要引入一大堆程序可能用到的组件。所以一个 helloworld 下来,相当巨大。而且很奇怪的事,以 QtWidgets/C++ 的方式使用了 QtWebEngine 的话,也会引入几乎所有的 QML 模块,那会儿一个简单的程序会达到一百多兆的大小!

Windows 平台还有一个特殊的问题。现在 Qt 的图形架构调用 OpenGL ES 2.0 进行底层的渲染,但刚安装完的 Windows 只支持到 OpenGL 1.1,针对这个问题,早期的 Qt 在 configure 阶段提供了两种选项,一是使用与 Chromium 一样的 ANGLE,把 OpenGL 调用翻译成 DirectX 调用,二者假定用户的系统安装了显卡驱动,能够支持 OpenGL 2.0。前者需要包含几个 ANGLE 的 DLL,直接又增加了 17M 的容量!

所幸这些事情现在都解决了。

首先,Qt 5 的早期版本已经提供了一个 configure 选项,可以不把 ICU 编译到 Qt 里面,但是官方下载版本仍然默认包含 ICU,后来 Qt 直接去掉了对 ICU 的依赖。现在的 Qt 在运行时找到 ICU 的 DLL 就会使用 ICU,否则调用系统的 API,如果都没有,就凑合着用 Qt 内置的算法处理国际化。

其次,QML 一些奇怪的依赖现在也去掉了。所以现在无论使用 QtWidgets 还是使用 QML,未压缩时都只有 20M 左右的容量。

最后一个,现在有多少用户的电脑是没有安装显卡驱动的呢?好吧,好像还真有,vmware 的虚拟机。但我觉得正常直接用系统自带的 OpenGL 驱动足矣。Qt 现在会在运行的时候自动探测 ANGLE 是否存在,如果存在,就使用 ANGLE,如果不存在,就使用系统自带的 OpenGL,不必像以前那样在 configure 阶段就配置好。另外,QtWidgets 没有使用 OpenGL,所以使用 QtWidgets/C++ 的程序也可以把 ANGLE 的 DLL 都去掉。

说这么多,具体怎么操作呢。非常简单,用 Qt 5.7 来说,现在提供了一个名为 windeployqt.exe 的程序,位于 Qt 的 bin 目录里面。运行这个程序,后面带程序名称就行了:

  1. 创建一个 dist 目录
  2. 把编译好的 qttest.exe 复制到 dist 目录。
  3. 打开 cmd.exe,cd 到 dist 目录
    4.1 运行 c:\qt5\bin\windeployqt.exe qttest.exe
    4.2 对于 QML 程序,需要指定 QML 目录: c:\qt5\bin\windeployqt.exe –qmldir d:\qttest qttest.exe

这时候就会看到 qt 已经把需要用到的 DLL 都复制过来了。我会在这个基础上再根据需要去掉一些东东:

  1. libEGL.dll, libGLESV2.dll 这两个文件是 ANGLE 的文件,可以去掉。opengl32sw.dll 是软件模拟 OpenGL,除非用户的系统连 DirectX 支持都不完整——虚拟机环境就是这样——不然这个文件也完全没有用。 QtWidgets/C++ 程序都不用 OpenGL,所以直接去掉即可。可在调用 windeployqt.exe 时加”–no-angle” 和 “–no-opengl-sw” 这两个参数。
  2. 如果没有使用 svg 的话,iconengines\qsvgicon.dll, imageformats\qsvg.dll, Qt5Svg.dll 这三个文件也可以删掉
  3. 如果没有国际化用户的话,translations 里面的翻译文件也可以删掉。
  4. QML 程序没有使用 QtWidgets/C++ 可以删掉 Qt5Widgets.dll
  5. 如果 imageformats 目录里面有几种图像格式没用上,也可以删掉。我自己通常把整个目录都删掉,Qt已经编译了 png 的支持,能读写程序包含的图标就够,其它格式不重要。
  6. qmltooling 和 Qt5Network.dll 是用于 QML 调试用的,可以删掉。

经过以上裁剪,使用 7zip 压缩完以后,一个 QtWidgets/C++ helloworld 程序最终只剩下 5.64M,一个 QtControls 程序是 5.86M, QtQuick 程序还会更小一些。

附注:

截止本文写作时,Qt 5.8 已经发布了。在这个版本里面,不再依赖于 OpenGL ES 2.0,而是绕过 ANGLE,直接调用 DirectX 渲染,到 Qt 5.10 还要直接支持 Vulkan API。不过目前仍然是试验性的,有兴趣的话可以看看 configure 的说明。

新的 Qt 5.8 重写了 configure 系统,能够更深入地裁剪 Qt。最小未压缩 5M 即可布署一个 QML 程序。

参考资料:

  1. Qt 5 ICU
    https://wiki.qt.io/Qt_5_ICU
    http://wiki.qt.io/Qt-5-QLocale

  2. Dynamically Loading Graphics Drivers
    http://doc.qt.io/qt-5/windows-requirements.html

  3. Qt Lite
    http://blog.qt.io/blog/2017/01/23/qt-5-8-released/



作者:时见疏星
链接:https://www.jianshu.com/p/0dd884a43de6
来源:简书
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

标签:简要,Qt,OpenGL,程序安装,程序,exe,QML,ICU
From: https://www.cnblogs.com/chinasoft/p/18402280

相关文章

  • Qt精品级项目——手撕信号槽机制的拳皇摇杆demo
    Qt精品级项目——手撕信号槽机制的拳皇摇杆demo1.前言&资源大家好,我是程序员Akgry。几天没见,阿克的Qt能力又是突飞猛进,阿克现在每天从睁眼学到闭眼,没了杂七杂八的东西掣肘,学起东西来非常舒适,虽然假期偷偷内卷很反人类,但是阿克今年就要参加秋招了,技术上的东西又岂敢怠慢。......
  • 【Qt】Qt与Html网页进行数据交互
    前言:此项目使用达梦数据库,以Qt制作服务器,Html制作网页客户端界面,可以通过任意浏览器访问。1、Qt与网页进行数据交互1.1、第一步:准备qwebchannel.js文件直接在qt的安装路径里复制即可1.2、第二步:在Qt的.pro文件加载webchannel组件在.pro文件添加如下组件:QT+=c......
  • Qt线程使用
    嗨嗨嗨,今天又学到了新的知识——线程,这个玩意在项目中使用的频率是非常高的,毕竟电脑的主线程就那么一个,那么这也就是我们为啥要学习线程的原因。比如说,我们们的游戏,如果我们的游戏界面显示的同时我们可以操作我们的装备、地图、人物移动等等这些都是线程做的,这样可以在一定程度......
  • Qt C++编程 从入门到实践 彭源 清华大学出版社
    第一章程序设计基础1.2.1输入和输出操作iostream叫做标准输入输出流库头文件namespacestd叫做标准命名空间cout、cin叫做标准输出、输入流对象有时候看见std::cout的代码,是因为没有事先声明cout对象是从标准命名空间调用的,::叫做域解析运算符,作用就是指明cout这个对象是......
  • Python教程(二十一) : 从零开始制作计算器应用【PyQt6】
    文章目录专栏列表环境准备代码解析主要组件初始化界面布局设置事件处理计算逻辑运行应用完整代码示例截图总结注意专栏列表Python教程(十):面向对象编程(OOP)Python教程(十一):单元测试与异常捕获Python教程(十二):面向对象高级编程详解Python教程(十三):常用内置模块详解Python......
  • 记录 PyQt6 / PySide 6 自定义边框窗口的 Bug 及可能可行的解决方案:窗口抖动和添加 DW
    前言:本篇文章将要讨论我在前不久发表的关于PyQt6/PySide6自定义边框窗口代码及内容中的问题:(终)PyQt6/PySide6+Pywin32自定义标题栏窗口+完全还原Windows原生窗口边框特效_pyside6win32无边框窗口-CSDN博客https://blog.csdn.net/2402_84665876/article/detail......
  • 坐牢第三十六天(QT)
    自定义QQ界面wedget.h#ifndefWIDGET_H#defineWIDGET_H#include<QWidget>#include<QDebug>//qt中信息调试类#include<QIcon>//图标类#include<QPushButton>//按钮类#include<QLabel>//标签类#include<QMovie>//动图类#include<QLineEd......
  • QT项目实战之音乐播放器2.0版本
    该版本相较于1.0版本最主要的不同在于连接数据库实现类似于歌曲收藏和取消收藏的功能。详细情况看我的这篇文章http://t.csdnimg.cn/WS5s8。效果展示VSMyMusicShow2.0define.hUseMySQL.h musicInfo.h VSMyMusicPlayer.h ......
  • 一套简约的qt 蓝色qss方案
    一套简约的qt蓝色qss方案直接使用qss代码QMenu{background:qlineargradient(x1:0,y1:0,x2:0,y2:1,stop:0rgba(255,255,255,240),stop:0.2rgba(255,255,255,200),stop:0.6rgba(255,255,255,160),stop:1rgba(255,255,255......
  • QT5 掌握debug调试的方法(简要内容:Memory查看内存地址的数值 和 查看变量值)(图文并茂)
    A1——选择构建模式(选项:debug调试、release发行、profile不知道…)A2——开始运行A3——开始调试(仅在debug调试模式下,断点调试助手才有效)A4——执行构建(生成输出目录及相应的文件,路径要求与工程的路径同级)A1——鼠标悬停变量名弹出,可固定窗口,Qt查看变量值的......