前往慧都网即下载Qt6最新试用版
一年半前,Qt做出了一个重大决定,开始使用CMake来构建Qt 6。做出此决定的主要原因是用户反馈。大多数Qt用户希望更轻松地将他们的Qt项目与其他软件集成在一起。根据当时的研究,CMake显然是Qt用户中最常用的构建工具-除了qmake。此外,迁移到CMake还为我们提供了摆脱内部构建工具维护负担的机会。
比决定更大的是迁移到CMake所需的工作。现在,基本的迁移工作已经完成,现在该分享我们的发现了。
尽管已被许多项目很好地建立和使用,但CMake缺少了先前Qt构建工具所支持的一些关键功能。这就是为什么我们与Kitware合作消除障碍并改善CMake的原因,从而使Qt项目和更大的CMake社区受益。
由于各种因素,构建Qt很复杂:
- Qt支持许多平台。
- Qt分为可独立构建或全部构建的模块。
- Qt支持不同的构建方式(前缀,非前缀,不同的功能集)
让我们看一下Qt的构建工具开关引起或影响的CMake改进。
Qt相关组件:
- QtitanRibbon: 遵循Microsoft Ribbon UI Paradigm for Qt技术的Ribbon UI组件,致力于为Windows、Linux和Mac OS X提供功能完整的Ribbon组件。
- QtitanChart :是一个C ++库,代表一组控件,这些控件使您可以快速地为应用程序提供漂亮而丰富的图表。并且支持所有主要的桌面操作系统。
- QtitanDataGrid :这个Qt数据网格组件使用纯C++创建,运行速度极快,处理大数据和超大数据集的效果突出。QtitanDataGrid完全集成了QtDesigner,因而极易适应其他相似的开发环境,保证100%兼容Qt GUI。
预编译头
在C ++项目中,可能会一遍又一遍地包含相同的头文件。对于库头尤其如此。如果工具链支持,则可以通过预编译头文件来加快编译速度。
在最长的时间内,CMake并未为此提供现成的支持。但是,在网上搜索代码片段以启用CMake中的预编译头的日子已经过去。从CMake 3.16开始,使用该target_precompile_headers命令可以添加要预编译的头文件列表。
Unity构建
另一种加快编译速度的方法是unity builds。这就是多名技术--它也被称为巨型构建、合并构建和单一编译单元。
这种技术创建了一个包含所有其他源文件的源文件,并且只编译这一个源文件。
#include "source_file1.cpp"
#include "source_file2.cpp"
/*...*/
#include "source_file8.cpp"
所有源文件的包含文件仅被处理一次,所有内容都以一个翻译单元结束,并且优化器对项目具有全局视图。
但是,并非每个C ++项目都可以不加修改地利用统一构建。
depfile支持AUTOMOC + Ninja
长期以来,人们一直在抱怨CMake的AUTOMOC,它会不必要地运行,或者在再次调用moc时无法正确检测到。原因之一是moc输出的确切依赖关系对于AUTOMOC是不可见的。
从Qt 5.15开始,moc学会了写出准确的文件,这些文件构成了moc输出的依赖性。CMake 3.17学会了读取moc的depfiles并将其用于Ninja生成器。
总结一下:使用Ninja生成器的Qt> = 5.15和CMake> = 3.17时,AUTOMOC知道正确的依赖项,可以在正确的时间重新运行moc。
Ninja Multi-Config
在Windows和macOS上,传统上Qt是用两种配置(调试和发布)构建的,但在一个构建目录中。
直到版本3.17引入了一个名为“ Ninja Multi-Config”的新生成器后,CMake才提供实现此目的的方法,该生成器可以一次构建多个配置。
iOS多架构构建
适用于iOS的Qt提供了模拟器和设备版本,该版本将针对实际目标构建的Qt与针对iOS模拟器构建的Qt相结合。尽管CMake 3.17引入了Ninja Multi-Config,但它的iOS支持尚未为iOS多体系结构构建做好准备。
CMake 3.18解决了这个问题,我们可以很高兴地为模拟器和设备进行构建。
文件(配置)
在Qt构建中,在配置时会生成大量带有动态创建内容的文件。configure_file但是,该命令仅使用输入文件,不使用字符串。
解决此问题的一种方法是让输入文件只有一个变量
--conf-file-content.txt.in--
@my-conf-file-content@
然后像这样调用configure_file
set(my-conf-file-content "This is the generated content!")
configure_file("my-conf-file-content.txt.in" "output.txt" @ONLY)
在CMake 3.18中,我们可以轻松实现相同目标:
file(CONFIGURE OUTPUT "output.txt" CONTENT "This is the generated content!")
评估!与功能同盟!
在Qt构建中,我们用尽了CMake语言作为实际编程语言执行的能力。不仅应该由用户调用CMake函数,而且在引擎盖下还有更复杂的设备。
让我们烦恼的一件事是,不可能动态调用函数。在qmake中,可以调用在变量中定义的函数,并且Qt5版本广泛使用此功能。
f = message
$${f}("Hello World!")
有多种方法可以使此工作在CMake中进行,包括生成一个随后包含的文件,但是特别是在Windows上,这将极大地减慢项目配置步骤。
CMake 3.18随附cmake_language(EVAL)用于评估CMake代码并cmake_language(CALL)调用宏或函数。
set(f "message")
cmake_language(CALL ${f} STATUS "Hello World!")
cmake_language(EVAL CODE "${f}(\"Hello World!\")")
稍后致电-延迟的代码
同样,qmake功能启发了CMake命令。
在QMake项目中,可以编写CONFIG += foo,然后在mkspecs/features/foo.prf处理实际的项目文件后加载。
这对于面向用户的API尤其有用。假设您正在为Android创建一个项目。
qt_add_executable(MyApp)
set_property(MyApp TARGET PROPERTY QT_ANDROID_EXTRA_LIBS SuperDuperLib)
qt_finalize_executable(MyApp)
该qt_finalize_executable调用将根据目标的属性为目标生成适当的部署设置。如果用户忘记致电qt_finalize_executable,则不会生成部署设置,也不会出现错误或警告。
但是,如果用户摆脱了打电话的负担,那会不会很棒qt_finalize_executable?在CMake 3.19中,Qt构建利用了新命令cmake_language(DEFER CALL)。这样一来,就可以在定义的时间调用函数,例如,在评估当前目录的项目文件之后。
不同目录范围的源文件上的属性
在某些地方,我们为模块创建目标,然后在子目录中将源文件添加到该目标。这样可以使源文件的名称与CMakeLists.txt文件保持接近。
但是,无法在这些位置指定每个源文件的属性-我们必须进入层次结构的顶层才能进行设置。我们并不经常这样做,但是有时在开发过程中会出现这种情况,以允许根据选项设置在编译时选择实验功能。
在CMake 3.18中,set_source_files_properties学会了在不同目录范围内设置属性:
该DIRECTORY选项获取指向处理后的源目录的路径的列表,并TARGET_DIRECTORY获取目标的列表,这些目标的源目录将用作设置源文件属性的作用域列表。
get_property()并且get_source_file_property()还具有相同的新参数,除了只能指定一个值而不是列表。
add_custom_command DEPFILE支持Makefile
CMake 3.20将为Unix Makefile生成器提供depfile支持。这也是将moc生成的depfile用于Makefile的前提。