异常本身几乎不带来运行时开销,只有在真正抛出时才显著影响性能。1. 异常机制依赖异常表和栈展开,编译期生成不影响正常流程;2. 抛出异常时需查找catch块、调用析构函数、执行catch逻辑,尤其是栈展开代价高;3. 错误码更轻量,适合频繁错误,但易遗漏且污染主逻辑;4. 建议将异常用于罕见情况,性能关键路径用错误码;5. 实际中可混合使用,结合可维护性与性能;6. 高性能环境如嵌入式系统可能禁用异常。总之,异常的开销集中在抛出阶段,适用于非常态错误。

C++的异常处理机制在现代编程中是一个很有争议的话题,尤其是在性能敏感的场景下。很多人关心:使用异常到底会带来多大的性能开销?和错误码相比,效率差异到底在哪?

简单来说:异常本身几乎不带来运行时开销,只有在真正抛出异常的时候才会显著影响性能。 也就是说,在“正常路径”上,异常机制对性能影响很小;但在异常触发的情况下,代价可能比错误码高很多。

异常处理的性能开销主要出现在哪?
C++的异常机制依赖于程序启动时建立的“异常表”和栈展开(stack unwinding)机制。这些结构在编译期生成,并不会直接影响正常执行流程的性能。但一旦抛出异常:
立即学习“C++免费学习笔记(深入)”;
- 程序需要查找匹配的
catch块; - 调用析构函数进行栈展开;
- 执行
catch中的逻辑。
这个过程比简单的返回错误码要复杂得多。尤其是栈展开,涉及大量运行时操作,比如调用每个局部对象的析构函数、检查异常处理程序等。

举个例子:
void func() {
std::vector v(10000);
// ...
if (error_condition) throw std::runtime_error("oops");
} 当异常被抛出时,系统必须安全地销毁 v,然后一层层向上找 catch。而如果只是返回错误码,这部分开销就完全不存在。
错误码 vs 异常:效率对比
从性能角度看,错误码通常更轻量:
✅ 优点:
- 直接通过返回值判断,几乎没有额外开销;
- 控制流清晰,适合频繁发生的错误情况。
❌ 缺点:
- 需要手动检查返回值,容易遗漏;
- 错误处理代码容易污染主逻辑;
- 多层嵌套调用时传递错误麻烦。
相比之下,异常虽然在正常路径上开销不大,但一旦抛出,代价远高于错误码。因此,如果你的应用中错误是“常态”而不是“例外”,那还是建议用错误码。
举个实际的例子:
- 在网络服务中,客户端请求参数错误很常见,这时候用错误码更合适;
- 而内存分配失败这种罕见情况,可以用异常来处理,不影响整体性能。
实际使用中的建议
结合性能与可维护性,可以考虑以下做法:
- ✅ 把异常用于真正的“异常”情况:比如资源不可用、逻辑错误等。
- ❌ 避免在高频路径中抛出异常:比如循环体内或每秒调用成千上万次的函数。
- ✅ 混合使用错误码与异常:关键性能路径用错误码,外围逻辑用异常简化代码。
- ✅ 评估编译器支持和优化能力:不同编译器对异常的支持程度不同,有些环境下开启异常会增加二进制体积和加载时间。
另外,如果你用的是嵌入式系统或者游戏引擎这类对性能要求极高的环境,很多团队会直接禁用异常处理。
基本上就这些。异常不是洪水猛兽,也不是银弹,关键是根据场景合理选择。性能影响只在异常真正抛出时显现,平时它几乎是透明的。











