C++20模块革新代码组织方式,提升编译效率。1. 模块通过export module声明,替代头文件包含;2. 使用import导入模块,避免重复解析;3. 支持分区管理,如Math:add和Math:mul;4. Clang/GCC需启用-fmodules编译;5. 可导入标准库模块如;6. 优势包括更快编译、更好封装,但需编译器与构建系统支持。

C++20 引入了模块(Modules),这是自 C++98 以来语言层面最重大的变革之一。它旨在替代传统的头文件包含机制,解决编译速度慢、命名冲突、宏污染等问题。使用模块可以显著提升编译效率,并让代码组织更清晰。
模块的基本概念
模块是一种新的编译单元,允许你将接口和实现分离,而无需依赖预处理器的 #include。一个模块可以导出函数、类、模板等,其他文件通过 import 来使用它,而不是包含头文件。
与传统头文件不同,模块只被编译一次,之后可被快速导入,避免重复解析。
创建一个简单模块
假设我们要创建一个名为 Math 的模块,提供一些数学函数。
立即学习“C++免费学习笔记(深入)”;
文件:Math.cppm(模块接口文件)
export module Math;export namespace math { int add(int a, int b) { return a + b; }
double multiply(double a, double b) { return a * b; }}
说明:
- module Math; 声明这是一个名为 Math 的模块。
- export 关键字用于导出能被外部访问的内容。
- 整个命名空间 math 被导出,其内部函数自动可访问。
导入并使用模块
在主程序中导入并使用该模块。
文件:main.cpp
import Math;include
int main() { std::cout << math::add(3, 4) << '\n'; std::cout << math::multiply(2.5, 4.0) << '\n'; return 0; }
注意:import Math; 直接导入模块,不需要引号或 .cppm 后缀。
模块分区(Module Partitions)
大型模块可以拆分为多个分区,便于管理。
例如,将 Math 模块拆分为加法和乘法两个部分。
文件:Math.add.cppm
module Math:add;export namespace math { int add(int a, int b) { return a + b; } }
文件:Math.mul.cppm
module Math:mul;export namespace math { double multiply(double a, double b) { return a * b; } }
文件:Math.cppm(主模块单元)
export module Math; export import :add; export import :mul;
主模块通过 export import :partition_name 导出分区内容。
编译模块(以 Clang 和 GCC 为例)
目前主流编译器对模块的支持逐步完善。以下是使用 Clang 17+ 或 GCC 13+ 的示例。
Clang 编译命令:
clang++ -std=c++20 -fmodules -xc++-system-header iostream # 预编译标准库模块(可选) clang++ -std=c++20 -fmodules Math.cppm -c -o Math.pcm clang++ -std=c++20 -fmodules main.cpp Math.cppm -o main
GCC 编译命令(实验性):
g++ -fmodules-ts -std=c++20 -c Math.cppm -o Math.o g++ -fmodules-ts -std=c++20 main.cpp -o main
GCC 的模块支持仍处于技术预览阶段,建议使用 Clang 或 MSVC 获取更好体验。
导入标准库模块
C++20 允许导入部分标准库作为模块(取决于编译器支持)。
例如,在 Clang 或 MSVC 中可以这样使用:
import; int main() { std::cout << "Hello, Modules!\n"; return 0; }
这比 #include iostream> 更高效,因为标准库模块只需编译一次。
模块的优势与注意事项
优势:
- 编译速度大幅提升,尤其在大型项目中。
- 避免宏和头文件的重复包含问题。
- 更好的封装性,未导出的内容对外不可见。
- 支持原子性编译,减少依赖传播。
注意事项:
- 并非所有编译器都完全支持 C++20 模块。
- 构建系统需适配模块编译流程(如 CMake 正在完善支持)。
- 模块文件通常使用 .cppm 扩展名,但不是强制的。
- 不能在模块中使用传统头文件混合导出(需谨慎过渡)。
基本上就这些。C++20 模块改变了我们组织代码的方式,虽然生态还在演进,但它代表了现代 C++ 的方向。从一个小模块开始尝试,逐步迁移到模块化编程是可行路径。











