Qt 子项目依赖管理:从原理到实践的最佳分析:depends还是 CONFIG += ordered
1. 问题背景
在Qt项目开发中,当一个工程包含多个子项目(如库、插件、测试模块)时,如何正确管理它们的构建顺序和依赖关系?
如:
在开发一个包含核心库(core)、GUI模块(gui)、插件(plugins)和测试(tests)的Qt项目时,我们发现:
• 如果plugins在core完成构建前启动编译,会导致链接失败
• 当tests同时依赖core和plugins时,手动管理顺序极易出错
# 典型构建错误示例
ld: cannot find -lcore # 核心库尚未编译完成
2. 为什么需要管理子项目依赖?
在复杂 Qt 项目中,模块通常存在依赖关系,例如:
• 插件(plugins) 依赖 核心库(core)
• 测试(tests) 依赖 主程序(app)和库(lib)
如果构建顺序错误,可能导致:
• 链接失败(未找到依赖库)
• 运行时错误(插件未正确初始化)
• 维护困难(新增模块时需手动调整顺序)
因此,Qt提供了两种方式管理依赖,但它们的适用场景不同。
常见的两种方式:
depends显式声明依赖CONFIG += ordered顺序构建
但,哪种方式更符合现代 Qt 开发的最佳实践?
3. depends:显式依赖声明(官方推荐)
3.1 基本语法
TEMPLATE = subdirs
SUBDIRS = core gui plugins tests# 显式声明依赖
gui.depends = core # gui 依赖 core
plugins.depends = core gui # plugins 依赖 core 和 gui
tests.depends = core # tests 仅依赖 core
3.2 优势
-
精准控制依赖
• 明确指定谁依赖谁,避免隐式顺序问题。
• 支持 非线性依赖(如多个插件依赖同一个库)。 -
可维护性高
• 新增模块时,只需添加depends,无需调整SUBDIRS顺序。
• 适合大型项目,模块化清晰。 -
官方明确推荐
“For complex dependencies between subprojects, use the
dependsvariable instead ofCONFIG += ordered.”
—— Qt 6 官方文档
3.3 适用场景
• 模块化项目(库 + 插件 + 测试)。
• 存在交叉依赖(如多个子项目依赖同一个核心模块)。
4. CONFIG += ordered:顺序构建(旧式方案)
4.1 基本语法
TEMPLATE = subdirs
CONFIG += ordered
SUBDIRS = core gui plugins tests # 严格按顺序构建:core → gui → plugins → tests
当新增database模块需要插入到core和gui之间时:
- 必须手动调整SUBDIRS顺序
- 可能破坏既有依赖关系
- 需要全面回归测试
4.2 存在问题
-
过于死板
• 必须确保SUBDIRS顺序完全匹配依赖关系,否则构建失败。
• 新增模块时需手动调整顺序,容易出错。 -
无法表达复杂依赖
• 如果tests依赖core,但plugins也依赖core,ordered无法直接表达。
4.3 残留用途
• 极简单的线性依赖项目(如 A → B → C)。
• 历史遗留代码维护。
5. 对比分析
| 特性 | depends | CONFIG += ordered |
|---|---|---|
| 依赖表达方式 | 显式声明(A.depends = B) | 隐式顺序(SUBDIRS 列表顺序) |
| 适用项目规模 | 中大型项目 | 极简单项目 |
| 维护成本 | 低(新增模块只需加依赖) | 高(需调整顺序) |
| 官方推荐度 | ✅ 推荐 | ⚠️ 不推荐 |
6. 示例
6.1 现代 Qt 项目推荐写法
TEMPLATE = subdirs
SUBDIRS = core utils gui plugins tests #(与顺序无关)# 显式声明依赖
utils.depends = core # utils 依赖 core
gui.depends = core utils # gui 依赖 core 和 utils
plugins.depends = gui # plugins 依赖 gui
tests.depends = core utils # tests 依赖 core 和 utils
6.2 遗弃用法
TEMPLATE = subdirs
CONFIG += ordered
SUBDIRS = core utils gui plugins tests # 依赖关系隐含在顺序中,难以维护
7. 结论
-
优先使用
depends
• 它是 Qt 官方推荐的方案,适合绝大多数项目。
• 提供更好的可读性、可维护性和灵活性。 -
避免
CONFIG += ordered
• 仅用于旧代码兼容或极其简单的线性构建。 -
保持依赖声明清晰
• 使用注释分组依赖,例如:# Core dependencies gui.depends = core plugins.depends = core gui
提示:在Qt Creator中,右键点击项目 → "Run qmake"可以实时验证依赖关系是否正确解析。
相关文章:
Qt 子项目依赖管理:从原理到实践的最佳分析:depends还是 CONFIG += ordered
1. 问题背景 在Qt项目开发中,当一个工程包含多个子项目(如库、插件、测试模块)时,如何正确管理它们的构建顺序和依赖关系? 如: 在开发一个包含核心库(core)、GUI模块(g…...
大数据专业学习路线
大数据专业学习路线 目录 基础知识核心技术进阶技能实战项目职业发展学习资源学习计划常见问题 1. 基础知识 1.1 编程语言 Python:大数据分析的基础语言 基础语法和数据类型函数和模块面向对象编程文件操作和异常处理常用库:NumPy, Pandas, Matplot…...
点云从入门到精通技术详解100篇-基于点云的三维多目标追踪与目标检测
目录 知识储备 基于Python和Open3D库实现的三维点云多目标检测与跟踪 技术要点解析 : 运行环境配置: 扩展改进建议 : 前言 三维多目标追踪技术 点云目标检测算法 2 二维多目标追踪框架及三维点云目标检测 2.1 二维多目标追踪框架 2.1.1 Deep SORT总体架构 2.1…...
dubbo配置中心
配置中心 简介 配置中心(config-center)在dubbo中可承担两类职责: 外部化配置:启动配置的集中式存储。流量治理规则存储。 Dubbo动态配置中心定义了两个不同层次的隔离选项,分别是namespace和group。 namespace&a…...
Java常用工具算法-5--哈希算法、加密算法、签名算法关系梳理
1、哈希算法 数学本质:将任意长度输入(明文)映射为固定长度输出(哈希值,也称摘要)。主要特点: 不可逆性:无法通过哈希值反推原始输入。雪崩效应:输入的微小变化导致哈希…...
python之安装PaddlePaddle和PaddleX解析pdf表格
目录标题 飞桨PaddlePaddle本地安装教程1-1. 基于 Docker 安装飞桨1-2. 基于 pip 安装飞桨2. 我两个环境 都选择的是pip 安装10. 如果报错10. 离线安装 飞桨PaddlePaddle本地安装教程 源码下载:https://github.com/PaddlePaddle/PaddleX/blob/release/3.0-beta1/do…...
【11408学习记录】英语语法核心突破:揭秘表语从句结构与通知写作实战技巧
表语从句与通知写作 英语语法总结——主从复合句表语从句语句结构系动词表语从句的位置 写作通知写作第二段第三段落款 每日一句词汇第一步:找谓语第二步:断句第三步:简化第一句第二句第三句第四句第五句 英语 语法总结——主从复合句 表语…...
React Native 0.79发布 - 更快的工具及更多改进
React Native 0.79版本发布了。 此版本在多个方面进行了性能改进,并修复了一些漏洞。首先,得益于延迟哈希技术,Metro的启动速度变快了,并且对包导出提供了稳定支持。由于JS包压缩方式的改变等原因,Android的启动时间也…...
封装红黑树实现map和set
前言: 之前我们学习了set与map容器的如何使用,红黑树的实现。接下来我们来看看如何通过封装红黑树,实现我们自己的set与map 相关文章:oi!让我来给你唠唠咋实现红黑树☝️-CSDN博客 超详细介绍map&…...
解码AI大脑:Claude的思维显微镜与语言炼金术
(前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站)。 一、多语言思维实验:Claude的“概念空间”如何运转? 跨语言谜题:反义词的…...
中科岩创基坑自动化监测解决方案
1.行业现状 城市基坑开挖具有施工风险高、施工难度大等特点。由于地下土体性质、荷载条件、施工环境的复杂性,单根据地质勘察资料和室内土工试验参数来确定设计和施工方案,往往含有许多不确定因素,对在施工过程中引发的土体性状、环境、邻近建…...
机器学习01-支持向量机(SVM)(未完)
参考浙大 胡浩基老师 的课以及以下链接: https://blog.csdn.net/m0_74100344/article/details/139560508 https://blog.csdn.net/2301_78630677/article/details/132657023 https://blog.csdn.net/lsb2002/article/details/131338700 一、一些定义 T是倒置&…...
CUDA编译器nvcc
nvcc(NVIDIA CUDA Compiler)是 NVIDIA 提供的 CUDA 编译器,用于编译 .cu 文件(CUDA C/C 代码)。它支持多种参数来控制编译过程,包括 GPU 架构优化、CUDA 库链接、调试选项等。以下是 nvcc 常用参数分类详解…...
Elasticsearch 系列专题 - 第一篇:Elasticsearch 入门
Elasticsearch 是一个功能强大的开源分布式搜索和分析引擎,广泛应用于日志分析、实时搜索、数据可视化等领域。本篇将带你了解 Elasticsearch 的基本概念、安装方法以及简单操作,帮助你快速上手。 1. 什么是 Elasticsearch? 1.1 Elasticsearch 的定义与核心概念 Elasticse…...
leetcode_数组 189. 轮转数组
189. 轮转数组 给定一个整数数组 nums,将数组中的元素向右轮转 k 个位置,其中 k 是非负数 示例 1: 输入: nums [1,2,3,4,5,6,7], k 3输出: [5,6,7,1,2,3,4] 示例 2: 输入:nums [-1,-100,3,99], k 2输出:[3,99,-1,-100] 思…...
Python基础全解析:从输入输出到字符编码的深度探索
一、Python程序交互的基石:Print函数详解 1.1 基础输出功能 # 输出数字 print(20.5) # 输出浮点数:20.5 print(0b0010) # 输出二进制数:10# 输出字符串 print(Hello World!) # 经典输出示例# 表达式计算 print(4 4 * (2-1)…...
[ctfshow web入门] web32
前置知识 协议相关博客:https://blog.csdn.net/m0_73353130/article/details/136212770 include:include "filename"这是最常用的方法,除此之外还可以 include url,被包含的文件会被当做代码执行。 data://:…...
指针数组 vs 数组指针
一、指针数组:「数组装指针」—— 每个元素都是指针 🔍 核心定义 语法:类型* 数组名[长度]; ([]优先级高于*,先形成数组,元素是指针)本质:一个 数组,数组的每个元素是 …...
鸿蒙开发中的并发与多线程
文章目录 前言异步并发 (Promise和async/await)多线程并发并发能力选择耗时任务并发执行场景常见业务场景 常驻任务并发执行场景常见业务场景 传统共享内存并发业务长时任务并发执行场景常见业务场景 并发任务管理线程间通信同语言线程间通信(ArkTS内)线…...
TCP和UDP的区别是什么?
1. 基本特性: TCP: 面向连接:在数据传输开始前,TCP需要在通信双方建立连接(三次握手)。可靠性:TCP保证数据的可靠传输,通过确认应答、重传机制、数据包顺序等确保数据无误到达。流量控制和拥塞…...
MySQL 函数(入门版)
目录 一、字符串函数 1、常用的字符串函数 2、函数演示 3、具体案例 二、数值函数 1、常用的数值函数 2、函数演示 3、具体案例 三、日期函数 1、常用的日期函数 2、函数演示 3、具体案例 四、流程函数 1、常用的流程函数 2、函数演示 3、具体案例 在MySQL中&a…...
Simulink中Signal Builder在新版中找不到怎么办
在较新的MATLAB版本中,新版Simulink中的Signal Builder用Signal Editor作为替代工具。 signal builder not shown in matlab - MATLAB Answers - MATLAB Central signalBuilderToSignalEditor 1.打开上面第二个链接 2.点击拷贝 3.然后在命令行中粘贴 4.然后就会…...
【补题】P10424 [蓝桥杯 2024 省 B] 好数(数位dp)
题意: 一个整数如果按从低位到高位的顺序,奇数位(个位、百位、万位……)上的数字是奇数,偶数位(十位、千位、十万位……)上的数字是偶数,我们就称之为“好数”。 给定一个正整数 N…...
SvelteKit 最新中文文档教程(19)—— 最佳实践之身份认证
前言 Svelte,一个语法简洁、入门容易,面向未来的前端框架。 从 Svelte 诞生之初,就备受开发者的喜爱,根据统计,从 2019 年到 2024 年,连续 6 年一直是开发者最感兴趣的前端框架 No.1: Svelte …...
Cursor编程-从入门到精通__0409
早期的Github Copilot 最近更新了,支持Agent编程,字节跳动Trae使用(免费),但成熟程度不如Cursor,Cursor前50次免费 Copilot VS Cursor*** 1,Cursor VSCode 二次开发,IDE级别 2&…...
VSCode、clangd、mingw 配置与使用
1.安装 安装如下软件: VSCodeclangd 扩展mingw-w64 2.配置 配置好 mingw-w64 到用户环境中。 在项目中设置 .clangd 扩展,设置 argument //setting.json"clangd.arguments": ["--query-driverD:\\Development\\Tools\\mingw64\\bin…...
深度学习处理文本(14)
使用Transformer进行序列到序列学习 正是序列到序列学习让Transformer真正大放异彩。与RNN相比,神经注意力使Transformer模型能够处理更长、更复杂的序列。要将英语翻译成西班牙语,你不会一个单词一个单词地阅读英语句子,将其含义保存在记忆中,然后再一个单词一个单词地生…...
核心案例 | 湖南汽车工程职业大学无人机操控与编队技术实验室
核心案例 | 湖南汽车工程职业大学无人机操控与编队技术实验室 为满足当今无人机行业应用需求,推动无人机技术的教育与实践深度融合,北京卓翼智能科技有限公司旗下品牌飞思实验室与湖南汽车工程职业大学强强联手,共同建设无人机操控与编队技术…...
Oracle 查看后台正在执行的 SQL 语句
在 Oracle 数据库中,要查看后台正在执行的 SQL 语句,可以通过查询动态性能视图(Dynamic Performance Views)或使用监控工具来实现。 1. 查询动态性能视图 (1) 查看当前活跃会话及其执行的 SQL 使用 v$session 和 v$sql 视图关联…...
SpringBoot整合MinIO快速入门:实现分布式文件存储与管理
文章目录 一、MinIO是什么?为什么选择它?1.1 什么是MinIO?1.2 核心优势 二、本地快速搭建MinIO服务2.1 Docker一键部署2.2 访问管理界面2.3 创建存储桶(Bucket) 三、SpringBoot集成MinIO客户端3.1 添加Maven依赖3.2 配…...
