C++进阶:1. 引用折叠规则

C++ 引用折叠规则

引用折叠(Reference Collapsing) 是 C++ 模板和 auto 类型推导中的核心隐式规则 ,专门解决多层引用嵌套 (如 T& &T&& &)的问题------C++ 语法不允许直接写"引用的引用" ,编译器会通过引用折叠将多层引用化简为单层引用

一、核心规则

只有 2 条铁律,所有嵌套引用最终都会被折叠成:

  1. 只要任意一层是左值引用(&),最终结果就是左值引用(&)
  2. 只有两层都是右值引用(&&),最终结果才是右值引用(&&)

简化表格:

嵌套写法 折叠结果 口诀
T& & T& 有左则左
T& && T& 有左则左
T&& & T& 有左则左
T&& && T&& 双右才右

注意:这是编译器自动执行 的规则,你不能手动写 T& &,但模板/auto 推导时会自动生成。

二、规则的诞生背景

C++11 引入右值引用后,模板/auto 推导会产生引用嵌套,例如:

cpp 复制代码
template<typename T>
void func(T&& param) { }  // 万能引用(Forwarding Reference)

当你传入左值/右值时,编译器会推导出嵌套引用,再触发折叠:

cpp 复制代码
int x = 10;
func(x);   // x 是左值 → T 推导为 int& → T&& = int& && → 折叠为 int&
func(10);  // 10 是右值 → T 推导为 int  → T&& = int&& → 无折叠,保持 int&&

三、最常见场景:万能引用 + 引用折叠

万能引用T&&/auto&&)的核心就是引用折叠,它能完美保留参数的左/右值属性:

1. 模板中的万能引用

cpp 复制代码
template<typename T>
void forward(T&& val) {
    // T&& 是万能引用,推导后触发引用折叠
}

int a = 5;
forward(a);      // 传入左值 int → T = int&
                 // 类型:int& && → 折叠为 int&(左值引用)

forward(10);     // 传入右值 int → T = int
                 // 类型:int&&(无折叠,右值引用)

2. auto 中的万能引用

cpp 复制代码
auto&& x = a;    // a 是左值 → auto 推导为 int&
                 // 类型:int& && → 折叠为 int&

auto&& y = 10;   // 10 是右值 → auto 推导为 int
                 // 类型:int&&

四、完整推导示例

我们直接看编译器的推导+折叠全过程:

cpp 复制代码
// 模板函数
template<typename T>
void test(T&& arg) {}

// 测试1:传入左值 int
int num = 20;
test(num);
// 推导步骤:
// 1. T 推导为 int&
// 2. T&& = int& &&
// 3. 引用折叠 → int&

// 测试2:传入右值 int
test(20);
// 推导步骤:
// 1. T 推导为 int
// 2. T&& = int&&
// 3. 无折叠 → int&&

五、关键作用:完美转发(std::forward)

引用折叠是 std::forward(完美转发) 的底层原理:

  • 它利用引用折叠规则,100% 保留参数的左值/右值属性
  • 让函数参数在传递过程中不丢失值类型,避免不必要的拷贝

示例:

cpp 复制代码
#include <utility>
template<typename T, typename... Args>
T create(Args&&... args) {
    // 完美转发:保留 args 的左/右值属性,靠引用折叠实现
    return T(std::forward<Args>(args)...);
}

六、总结(必背)

  1. 规则:有左则左,双右才右
  2. 触发场景 :模板T&&auto&&(万能引用)的类型推导
  3. 核心目的 :解决"引用的引用"语法非法问题,实现完美转发
  4. 结果 :所有嵌套引用最终只会是 &&& 两种之一

总结

  1. 引用折叠是编译器自动化简嵌套引用的隐式规则;
  2. 核心口诀:有左则左,双右才右
  3. 它是万能引用 和**std::forward 完美转发**的底层基础。
相关推荐
404号扳手1 小时前
Java 进阶知识(七)
java·后端
酬勤-人间道1 小时前
VTK 与 Cesium-native 结合实践:小场景三维编辑 + 数字地球精准贴合
c++·qt·vtk·遥感·岩土·cesium-native
智者知已应修善业1 小时前
【51单片机8个LED的花样12亮34熄56间隔78闪烁3秒3闪烁】2023-11-4
c++·经验分享·笔记·算法·51单片机
小马爱打代码1 小时前
Spring框架:介绍和快速入门
java·后端·spring
糖果店的幽灵1 小时前
LangChain 1.3 完全教程:从入门到精通-Part 7: Documents(文档处理)
java·python·langchain
初中就开始混世的大魔王1 小时前
5 Fast DDS-Discovery
网络·c++·算法·中间件
Deep-w1 小时前
【MATLAB】基于模型预测控制的自适应巡航车辆过渡工况安全控制研究
开发语言·人工智能·算法·机器学习·matlab
Java_2017_csdn1 小时前
Java 策略模式(Strategy Pattern)-(三)
java·开发语言·servlet
许彰午1 小时前
06_Java面向对象入门
java·开发语言·python