如何使用C++20 Modules减少大型项目的构建时间? (物理依赖解耦)
技术百科
裘德小鎮的故事
发布时间:2026-01-21
浏览: 次 include是构建瓶颈,因每个.cpp文件重复解析同一头文件,导致预处理、宏展开、模板实例化全量重做,无法有效缓存;Modules通过二进制模块接口(.pcm/.ifc)避免重复解析,实现物理依赖解耦。
为什么 #include 是构建瓶颈?
传统头文件包含机制会让每个 .cpp 文件重复解析同一份头文件(比如 、),哪怕只用其中一两个符号。预处理器展开、宏重定义、模板实例化全得重做一遍——这直接导致编译器无法有效缓存,且并行编译时大量重复工作。
Modules 把接口与实现分离成二进制模块单元,编译一次后生成模块接口单元(.pcm),后续导入只需读取该二进制快照,跳过词法/语法分析和宏处理。
如何导出一个可复用的模块接口?
模块接口单元(.ixx 或 .cppm)必须显式声明导出内容,不能靠“包含即导出”。未导出的实现细节(如辅助函数、私有类)不会污染导入者的编译环境,这是物理依赖解耦的关键。
-
export module math_utils;声明模块名,必须是文件首条非注释语句 - 仅
export修饰的声明才对外可见:export int add(int a, int b) { return a + b; } - 内部实现可写在
module;区块里(不导出):module; namespace detail { constexpr int square(int x) { return x * x; } } - 可导出整个命名空间:
export namespace geometry { struct Point { int x, y; }; }
Clang / MSVC 模块构建流程差异
Clang 和 MSVC 对模块的构建链路设计不同,直接影响 CI 配置和增量编译行为:
- Clang 要求先用
-x c++-system-header或--precompile预编译标准库模块(如std),再通过-fpre引用;否则会回退到头文件模式
built-module-path
- MSVC 默认启用标准库模块支持(需 `/std:c++20 /experimental:module`),且模块接口编译命令为
cl /c /interface /exportHeader,生成.ifc文件 - 两者都不支持跨编译器模块二进制兼容——
clang-built .pcm不能被cl.exe导入 - 模块依赖必须显式声明:
import std.core;或import "my_math.ixx";,路径需在-fmodule-map-file(Clang)或 `/module:reference`(MSVC)中注册
哪些代码不适合立刻模块化?
不是所有头文件都能平滑迁移。以下情况会卡住或倒退:
- 含复杂宏逻辑的头文件(如 Boost.Preprocessor 或 Qt 的
MOC宏)——Modules 不解析宏,export macro尚未标准化 - 使用
#pragma once或#ifndef包裹但实际依赖文本包含顺序的头文件(如某些 C 风格 SDK)——Modules 消除了包含顺序语义 - 模板定义分散在多个头中、靠隐式实例化传播的代码——需确保所有模板声明在模块接口中完整导出,否则链接时报
undefined reference - 第三方库未提供模块接口(如大多数 vcpkg 包)——只能用
module : partition封装其头文件,但无法消除预处理开销
真正节省时间的模块化,始于对“谁依赖谁”的显式建模,而不是把 #include 换成 import 就完事。模块接口文件一旦变更,所有导入它的 TU 都要重编译——这点比头文件更严格,别低估接口稳定性成本。
# ai
# 这是
# 多个
# 都能
# 都要
# 只需
# 会让
# 都不
# 一遍
# mac
# c++
# int
# 标准库
# 接口
# 为什么
# Interface
# 封装
# 头文件
# 命名空间
# Struct
# map
# 处理器
# Namespace
# include
# undefined
# 重做
# qt
# 预处理器
相关栏目:
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
AI推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
SEO优化<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
技术百科<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
谷歌推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
百度推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
网络营销<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
案例网站<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
精选文章<?muma echo $count; ?>
】
相关推荐
- php转mp4怎么保留字幕_php处理带字幕视频转
- Windows 11如何开启文件夹加密(EFS)_
- 用Python构建微服务架构实践_FastAPI与
- Win11怎么关闭OneDrive同步_Win11
- Python lxml的etree和Element
- 如何在Golang中捕获HTTP服务器错误_Gol
- 如何使用Golang实现容器安全扫描_Golang
- Win11怎么开启空间音效_Windows11耳机
- 如何使用Golang table-driven f
- 如何使用Golang安装依赖库_管理模块和第三方包
- php下载安装包太大怎么下载_分卷压缩下载方法【教
- Go 中实现 Python urllib.quot
- Win10怎样安装Excel数据分析工具_Win1
- Win11怎么设置开机密码_Windows11账户
- 作用域操作符会影响性能吗_php静态调用性能分析【
- Python邮件系统自动化教程_批量发送解析与模板
- c++如何判断文件是否存在_c++ filesys
- LINUX的SELinux是什么_详解LINUX强
- 如何在 Laravel 中通过嵌套关联关系进行 o
- 如何使用Golang指针与结构体结合_修改结构体内
- 如何使用Golang实现容器自动化运维_Golan
- Win11如何隐藏桌面图标 Win11一键隐藏/显
- Windows10电脑怎么设置防火墙出站规则_Wi
- Mac怎么设置登录项_Mac管理开机自启动程序【教
- MAC如何修改默认应用程序_MAC文件后缀关联设置
- C#怎么使用委托和事件 C# delegate与e
- Win11怎样安装搜狗输入法_Win11安装搜狗输
- c++ std::future和std::prom
- PHP中require语句后直接调用返回对象方法的
- LINUX如何删除用户和用户组_Linux use
- Linux如何挂载新硬盘_Linux磁盘分区格式化
- PHP cURL GET请求:正确设置请求头与身份
- 如何将文本文件中的竖排字符串转换为横排字符串
- Win10如何关闭安全中心所有通知 Win10禁用
- Python与MongoDB NoSQL开发实战_
- Win11任务栏颜色怎么改_Win11自定义任务栏
- 如何在 Go 中正确初始化结构体中的 map 字段
- c++如何用AFL++进行模糊测试 c++ Fuz
- Win11怎么设置右键刷新选项_Windows11
- Django 测试数据库表缺失与字段未创建问题的完
- php485返回数据不完整怎么办_php485数据
- Win11时间不对怎么同步_Win11自动校准互联
- Win11怎样安装钉钉客户端_Win11安装钉钉教
- 如何使用Golang匿名函数_快速定义临时函数逻辑
- php本地部署后数据库连接报错_1045acces
- PowerShell怎么创建复杂的XML结构
- Win10系统怎么查看显卡温度_Win10任务管理
- Win11如何关闭游戏模式 Win11禁用Xbox
- Win11如何设置环境变量 Win11添加和修改系
- Win10如何更改开机密码_Windows10登录


QQ客服