两者语义一致但编译器专用:[[msvc::forceinline]]仅MSVC识别,[[gnu::always_inline]]仅GCC/Clang识别;实际常用__forceinline和__attribute__((always_inline)),且强制内联易引发代码膨胀、调试困难等副作用。
两者语义一致,都是要求编译器必须内联函数,但仅对各自目标编译器生效:[[msvc::forceinline]] 只被 MSVC 识别,[[gnu::always_inline]] 只被 GCC/Clang(GNU 模式)识别。
MSVC 完全忽略 [[gnu::always_inline]],GCC/Clang 则无视 [[msvc::forceinline]]。如果写成:
[[msvc::forceinline]] [[gnu::always_inline]] void foo() { }
在 MSVC 下只认第一个,在 GCC 下只认第二个——但不会报错,也不会叠加效果。实际项目中常需条件编译:
_MSC_VER 检测 MSVC__GNUC__ 或 __clang__ 检测 GNU/ClangMSVC 的 [[msvc::forceinline]] 会绕过大多数内联限制(如递归、含 try/catch、过大函数体),只要语法合法就尝试强制展开;而 GCC 的 [[gnu::always_inline]] 虽也尽力,但遇到某些结构(如变长数组、嵌套 lambda 捕获复杂对象)可能静默降级为普通 inline,并给出警告(-Winline 可捕获)。
[[gnu::always_inline]] 的处理更接近 GCC,但某些版本对模板实例化内联更保守
标准属性是 C++20 引入的,但实际工程中仍大量使用传统关键字:
__forceinline void foo() { },比 [[msvc::forceinline]] 兼容性更好(支持旧版 MSVC 2015+)__attribute__((always_inline)) void foo() { },Clang 也完全支持,且比 [[gnu::always_inline]] 解析更稳定[[msvc::forceinline]])强制内联最常被忽视的问题不是语法,而是它放大了以下风险:
-flto 本身会重做内联决策,此时 [[gnu::always_inline]] 可能被覆盖或冲突template [[gnu::always_inline]] void bar(T) ,每个 T 都生成一份内联副本,而非共享一份可复用代码除非 profiler 明确指出某处函数调用开销占比显著,否则别轻易加强制内联——编译器自己的 inline heuristics 在 -O2/-O3 下通常更靠谱。