[[likely]]和[[unlikely]]是C++20引入的分支预测属性,用于提示编译器某分支的执行概率以优化性能。它们应用于if、switch等语句前,帮助CPU流水线更准确预测执行路径,减少因预测错误导致的性能损失。例如错误处理用[[unlikely]]标记,主流程用[[likely]]提示。该特性需C++20支持,如GCC 10+或Clang 12+,且仅作优化建议,编译器可忽略。正确使用可在高频代码中提升效率。
在 C++20 中,[[likely]] 和 [[unlikely]] 是两个新的属性(attributes),用于向编译器提示某个代码分支的执行概率,帮助编译器优化生成的机器码,提升程序性能。它们主要用于条件语句中的分支预测优化。
现代 CPU 使用流水线和分支预测技术来提高指令执行效率。当遇到 if-else 或 switch 这类分支结构时,如果 CPU 能“猜中”将要执行的分支,就能提前加载并执行后续指令。若预测错误,则需要清空流水线,造成性能损失。
通过使用 [[likely]] 和 [[unlikely]],程序员可以显式告诉编译器哪条路径更可能被执行,从而让编译器将高频路径的代码放在更有利于执行的位置(例如减少跳转、优化缓存局部性等)。
这两个属性用在语句前,通常配合 if、switch 或循环使用:
if (condition) [[likely]] {
// 预期该分支大概率会执行
}
if (error) [[unlikely]] {
// 错误处理等小概率事件
handle_error();
}
也可以用于 switch 的 case 标签:
switch (state) {
case OK: [[likely]]
process_ok();
break;
case ERROR: [[unlikely]]
log_error();
break;
}
例如:
bool parse_data(const char* data, size_t len) {
if (!data || len == 0) [[unlikely]] {
return false; // 输入非法属于异常情况
}
// 正常解析流程
for (size_t i = 0; i zuojiankuohaophpcn len; ++i) [[likely]] {
process(data[i]);
}
return true;}
基本上就这些。正确使用 [[likely]] 和 [[unlikely]] 可以在关键路径上带来可观的性能提升,尤其是在高频调用的底层库或系统代码中。虽然日常应用中影响可能不大,但在追求极致性能的场景下,这是一个简单有效的优化手段。