预编译头(PCH)通过提前将稳定、频繁包含的头文件编译为二进制格式,避免重复解析与处理,显著提升编译速度,实测可减少30%–70%编译时间;需正确配置入口头文件、编译选项及包含方式,并规避宏不一致、频繁修改头文件、命名空间污染等常见问题。

预编译头(Precompiled Header,简称 PCH)是 c++ 工程中一种用来显著加快编译速度的机制,核心思想是:把项目中**稳定、频繁包含、几乎不改动**的头文件(比如 <vector></vector>、<String></string>、<memory></memory>、mfc 或 qt 的大头文件等)提前编译成二进制中间格式,后续每个源文件(.cpp)就不用重复解析和编译这些内容,直接“加载”即可。
为什么 PCH 能提速?
一个大型工程里,每个 .cpp 文件往往都包含几十个标准库或框架头文件。而这些头文件本身又层层 #include,导致重复解析语法树、宏展开、模板实例化前处理等工作——非常耗时。PCH 把这部分“公共开销”一次性做完,之后所有使用它的源文件跳过这步,实测可减少 30%–70% 的整体编译时间(尤其在增量编译时效果更明显)。
怎么正确启用 PCH?
不同编译器略有差异,但通用逻辑一致:
- 选一个头文件作为 PCH 入口(如
StdAfx.h或pch.h),只在这里集中 include 稳定头文件;不要放项目私有头、条件宏、内联函数定义 - 单独编译这个头文件:用编译器选项生成 PCH 文件(如 MSVC 用
/Yc"pch.h",Clang/GCC 用-x c++-header -o pch.h.gch) - 其他 .cpp 文件开头第一行必须是
#include "pch.h"(MSVC 要求严格;Clang/GCC 可通过-include pch.h绕过位置限制) - 编译时开启 PCH 使用(如 MSVC 加
/Yu"pch.h",Clang/GCC 加-include pch.h并确保 .gch 文件在搜索路径中)
PCH 使用常见坑点
用不好反而拖慢编译,甚至引发隐蔽错误:
立即学习“C++免费学习笔记(深入)”;
- pch.h 里不能出现 #ifdef/#ifndef 控制的头文件——因为 PCH 是独立编译的,宏状态和实际 .cpp 不一致会导致行为错乱
- 不要把经常修改的头文件放进 PCH——一旦它变,整个 PCH 失效,所有源文件被迫全量重编译
- 避免在 PCH 中 using Namespace std;——污染全局命名空间,易引发冲突,且违背模块隔离原则
- 跨平台项目注意 PCH 文件不可移植(.pch/.gch 格式依赖编译器+平台+ABI),CI 构建需清理或禁用
除了 PCH,还有哪些实用提速技巧?
PCH 是“大招”,但日常优化更靠细节:
- 用 #include
替代 #include “xxx” ——让编译器跳过本地路径扫描,优先查系统/标准路径 - 前置声明(forward declaration)代替 #include——尤其在头文件中,对只用指针/引用的类,用
class Foo;即可 - 拆分大头文件,按需引入——比如不把整个
<boost></boost>塞进 PCH,只在真正需要序列化的 .cpp 里包含 - 启用编译器缓存(ccache / clcache)——对未变的 .cpp,直接复用上次编译结果,比 PCH 更细粒度
基本上就这些。PCH 不是银弹,但搭配合理的头文件管理、构建缓存和模块化设计,能让中大型 C++ 工程的编译体验从“喝咖啡等待”变成“顺手敲几行就完事”。