谈谈 Qt 程序安装包的大小,以及简要打包指南
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 目录里面。运行这个程序,后面带程序名称就行了:
- 创建一个 dist 目录
- 把编译好的 qttest.exe 复制到 dist 目录。
- 打开 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 都复制过来了。我会在这个基础上再根据需要去掉一些东东:
- libEGL.dll, libGLESV2.dll 这两个文件是 ANGLE 的文件,可以去掉。opengl32sw.dll 是软件模拟 OpenGL,除非用户的系统连 DirectX 支持都不完整——虚拟机环境就是这样——不然这个文件也完全没有用。 QtWidgets/C++ 程序都不用 OpenGL,所以直接去掉即可。可在调用 windeployqt.exe 时加”–no-angle” 和 “–no-opengl-sw” 这两个参数。
- 如果没有使用 svg 的话,iconengines\qsvgicon.dll, imageformats\qsvg.dll, Qt5Svg.dll 这三个文件也可以删掉
- 如果没有国际化用户的话,translations 里面的翻译文件也可以删掉。
- QML 程序没有使用 QtWidgets/C++ 可以删掉 Qt5Widgets.dll
- 如果 imageformats 目录里面有几种图像格式没用上,也可以删掉。我自己通常把整个目录都删掉,Qt已经编译了 png 的支持,能读写程序包含的图标就够,其它格式不重要。
- 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 程序。
参考资料:
Qt 5 ICU
https://wiki.qt.io/Qt_5_ICU
http://wiki.qt.io/Qt-5-QLocaleDynamically Loading Graphics Drivers
http://doc.qt.io/qt-5/windows-requirements.html
作者:时见疏星
链接:https://www.jianshu.com/p/0dd884a43de6
来源:简书
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
谈谈 Qt 程序安装包的大小,以及简要打包指南的更多相关文章
- NSIS控制面板中显示安装包的大小和禁止多个安装程序实例
转载:http://www.yhxs3344.net/jscript/nsis 转载:http://www.yhxs3344.net/archives/1292 1.控制面板中显示安装包的大小 ;需要 ...
- 减少iOS应用程序安装包大小
安装包优化大小方法: <资源优化> 1.去除无用资源 通过几次项目的升级后,项目中会出现一些没有用到的图片.这些图片在我们导入到项目中后,之后项目升级过程后并没有再次用到. 那这些图片我们 ...
- Mac OS平台下应用程序安装包制作工具Packages的使用介绍
一.介绍 Windows下面开发好的应用程序要进行分发时有很多打包工具可供选择,如Inno Setup, InstallShield, NSIS, Advanced Installer, Qt Ins ...
- 基于DevExpress的Winform程序安装包的制作
在我们做系统开发的时候,都会面临一个安装包制作的问题,如何把我们做好的系统,通过安装包工具整合成一个安装包给客户进行安装.安装包的优势就是一步步安装就可以了,不用复制一大堆文件给客户,还怕缺少那个文件 ...
- 使用WinRar软件制作程序安装包
之前我写过使用好压软件打包程序,见随笔: 使用好压(HaoZip)软件打包EverEdit制作安装程序 - Fetty - 博客园http://www.cnblogs.com/fetty/p/4907 ...
- 用inno Setup做应用程序安装包的示例脚本(.iss文件)(
用innoSetup做应用程序安装包的示例脚本(.iss文件),具体要看innoSetup附带的文档,好象是pascal语言写的脚本. 示例1(应用程序.exe,客户端安装): ;{089D6802- ...
- 【原创】VB6.0应用程序安装包的生成(Setup Factory 9.0制作安装包的方法)
VB6.0应用程序安装包的生成,利用其自带的打包工具生成的安装程序很简陋,一点不美观:如果想让自己的应用程序安装的时候显得高大上一点,本教程提供使用Setup Factory 9.0制作安装包的方法. ...
- 制作Linux下程序安装包——使用脚本打包bin、run等安装包
制作简单的安装包的时候可以简单的用cat命令连接两个文件,然后头部是脚本文件,执行的时候把下面的文件分解出来就行了.一般这个后部分的文件是个压缩 包,那样,就能够打包很多文件了,在脚本中解压出来即可. ...
- php实现在线下载程序安装包功能
在线下载程序安装包可以很方便在服务器端下载各种程序安装包(Discuz!.phpwind.Dedecms.WordPress....等一些常用程序)并存储在服务器,大大减少站长上传程序安装包时间.默认 ...
- C#软件winform程序安装包制作及卸载程序制作
使用vs2010 winform程序开发的软件的人比较多,程序的开发是为了在不同的人不同的机器使用,为了使不同的机器能使用该软件就需要在制作程序安装包,安装包里必须包含该软件运行所选的所有环境,下面就 ...
随机推荐
- [CRCI2008-2009] CVJETICI
[CRCI2008-2009] CVJETICI 观察图片及样例一: 注:下文中的被占领,指的是在这一个区间内,才有交叉开花的可能. 第一张小图发现 $2 \sim 3$ 被占领. 第二张小图发现 $ ...
- DeepSeek MOE 代码实现
前置知识: PyTorch 基础函数操作整理 1. topk 操作 功能: torch.topk 用于返回输入张量中指定维度上的前 k 个最大元素及其对应的索引. 示例代码: import torch ...
- 基于人类反馈的强化学习 RLHF
1.强化学习和语言模型的联系 agent: 语言模型本身 state: prompt(input tokens) action: 选择哪个token作为下一个token(贪婪,top k,top p) ...
- C# 异步编程:从 async/await 到 Task 并行库的全面解析
引言 在现代软件开发中,处理高并发和耗时操作是一个常见的挑战.C# 提供了强大的异步编程模型,它允许程序在执行耗时操作时不会阻塞主线程,从而提高程序的响应性和性能.其中,async/await 关键字 ...
- 性能提升30%!袋鼠云数栈基于 Apache Hudi 的性能优化实战解析
Apache Hudi 是一款开源的数据湖解决方案,它能够帮助企业更好地管理和分析海量数据,支持高效的数据更新和查询.并提供多种数据压缩和存储格式以及索引功能,从而为企业数据仓库实践提供更加灵活和高效 ...
- 技术干货|如何利用 ChunJun 实现数据实时同步?
实时同步是 ChunJun 的⼀个重要特性,指在数据同步过程中,数据源与⽬标系统之间的数据传输和更新⼏乎在同⼀时间进⾏. 在实时同步场景中我们更加关注源端,当源系统中的数据发⽣变化时,这些变化会⽴即传 ...
- springboot~入门第二篇~页面html跳转~
遇到的问题:按照别人的blog搭jsp页面就是html页面跳转不了,总是如图: 终于找到了一个能用的blog ,换 thymeleaf(html页面跳转)成功. 控制器代码 注意下: @Control ...
- stm32主要用来做什么?
STM32主要用来做什么?一个从机械转行的十年老兵血泪经验 写在前面:一个改变命运的小芯片 说起STM32,我真的是百感交集. 十年前,我还是个刚从某211大学机械专业毕业的愣头青,对嵌入式.单片机这 ...
- mysql递归查询(父级,子集)
①查询父级 表: 先给表创建函数: 1 CREATE FUNCTION `getParentList`(rootId varchar(100)) 2 RETURNS varchar(1000) 3 B ...
- linux chmod 修改移动硬盘无效
转载 http://blog.csdn.net/lanyang123456/article/details/47683351 linux 下挂载windows ntfs 硬盘,采用开机挂载,修改/et ...