【Effective Modern C++】第五章 右值引用、移动语义和完美转发:24. 区分万能引用和右值引用

T&&有两种身份,并非只有右值引用

  • 我们默认T&&是右值引用,但事实是,T&&在源码中可能有两种完全不同的含义;
  • 第一种:真正的右值引用 ------ 只能绑定到右值(临时对象、std::move后的对象),核心作用是识别可移动的对象;
  • 第二种:通用引用 (也叫转发引用)------ 本质是 "二重性引用",源码上写的是T&&,但可以表现为左值引用或右值引用,能绑定几乎任何东西(左值 / 右值、const/non-const、volatile 等);
  • 通用引用是一种 "抽象"(而非底层真相),底层是 "引用折叠",但这种抽象足够实用,能帮我们快速区分两种引用。

通用引用的判断:必须同时满足两个条件

1:存在类型推导
  • 要么是函数模板形参T(或模板参数)需要在函数调用时自动推导(排除调用者显式指定类型的边缘情况);
  • 要么是auto&& 声明的对象auto的类型需要在初始化时自动推导;
  • 没有类型推导,就不可能是通用引用。
2:声明形式必须是纯 T&&
  • 引用的声明必须严格是 "T&&",不能有任何额外修饰(比如constvolatile,或嵌套类型如std::vector<T>&&);
  • 这里的T只是模板参数名,换成其他名字(如MyTemplateType&&)也可以,关键是 "类型名 +&&" 的纯形式。
符合条件的通用引用示例
c++ 复制代码
// 1. 函数模板形参:有类型推导+纯T&& → 通用引用
template<typename T>
void f(T&& param);

// 2. auto&&声明:有类型推导+纯&& → 通用引用
auto&& var2 = var1; // var1是左值,var2就变成左值引用;var1是右值,就变成右值引用

// 3. 模板参数名非T,仍是通用引用(纯形式+推导)
template<typename MyTemplateType>
void someFunc(MyTemplateType&& param);

右值引用的判断:缺一个条件,就是右值引用

只要不满足通用引用的两个必备条件,哪怕源码是T&&,本质也是右值引用 ------ 只能绑定右值,不能绑定左值。

右值引用的示例
c++ 复制代码
// 1. 无类型推导:Widget&& 固定类型 → 右值引用
void f(Widget&& param);
Widget&& var1 = Widget(); // 无推导,直接绑定临时对象(右值)

// 2. 有类型推导,但不是纯T&&(带容器)→ 右值引用
template<typename T>
void f(std::vector<T>&& param); // 形式是std::vector<T>&&,不是纯T&&

// 3. 有类型推导,但不是纯T&&(带const)→ 右值引用
template<typename T>
void f(const T&& param); // 带const修饰,不是纯T&&
  • 向以上右值引用传递左值,会直接编译报错(右值引用不能绑定左值),比如std::vector<int> v; f(v);会报错。

模板中的T&&,不一定是通用引用

很多人误以为 "模板里的T&&都是通用引用",其实不然,核心看 "是否有独立的类型推导"。

错误示例:std::vector::push_back
c++ 复制代码
// vector的push_back成员函数
template<class T, class Allocator = allocator<T>>
class vector {
public:
    void push_back(T&& x); // 看似T&&,但不是通用引用
};
  • 原因:push_backT,是vector的模板参数 ------ 当vector<Widget> v实例化后,T已经确定是Widgetpush_back的声明就变成void push_back(Widget&& x)没有类型推导
  • 结论:push_backT&&右值引用 ,不能传递左值(比如Widget w; v.push_back(w);会报错)。
正确示例:std::vector::emplace_back
c++ 复制代码
template<class T, class Allocator = allocator<T>>
class vector {
public:
    template <class... Args>
    void emplace_back(Args&&... args); // Args&&是通用引用
};
  • 原因:emplace_backArgs,是独立于vector的模板参数 ,每次调用emplace_back时,Args都会重新推导;
  • 结论:Args&&满足 "类型推导 + 纯形式",是通用引用,可以绑定任何类型、任何值类别的实参。

auto&&:容易被忽略的通用引用

auto&&声明的对象,也是通用引用(满足 "类型推导 + 纯 &&"),虽然不如模板函数中的通用引用常见,但在 C++11/14 中高频出现。

示例
c++ 复制代码
// C++11:auto&&绑定左值/右值,会自动"变身"
auto&& var = 10;        // 绑定右值 → 变成int&&(右值引用)
auto&& var2 = var;      // 绑定左值 → 变成int&(左值引用)

// C++14:lambda表达式中的auto&&(高频用法)
auto timeFuncInvocation =
    [](auto&& func, auto&&... params) // func和params都是通用引用
    {
        // 转发参数,保留原始值类别(Item33详细讲解)
        std::forward<decltype(func)>(func)(
            std::forward<decltype(params)>(params)...
        );
    };
  • auto&&可以绑定任何对象(左值 / 右值),其最终类型(左值引用 / 右值引用),由初始化时的实参值类别决定 ------ 和模板函数中的通用引用 "变身" 逻辑完全一致。

通用引用的 "抽象" 本质

通用引用的概念,是一种 "抽象"(而非底层真相),底层实现是 "引用折叠"。但这种抽象非常实用:

  • 不用深究引用折叠的细节,就能快速区分T&&的身份;
  • 方便阅读代码(判断T&&是只绑定右值,还是能绑定任何对象);
  • 方便交流(明确告知合作者 "这里用的是通用引用,而非右值引用")。

总结

  1. T&&有两种身份:右值引用,或通用引用(转发引用);
  2. 通用引用的判断标准(缺一不可):① 存在类型推导(模板推导 /auto 推导);② 声明形式是纯T&&(无任何修饰);
  3. 不满足以上两个条件,T&&就是右值引用 ------ 只能绑定右值,不能绑定左值;
  4. 通用引用的 "变身" 特性:被右值初始化→变成右值引用;被左值初始化→变成左值引用;
  5. 模板中的T&&不一定是通用引用(如vector::push_back),关键看是否有 "独立的类型推导";
  6. auto&&也是通用引用,可绑定任何对象,常见于 C++14 lambda 表达式。

原著在线阅读地址

相关推荐
hetao173383730 分钟前
2026-04-09~12 hetao1733837 的刷题记录
c++·算法
6Hzlia32 分钟前
【Hot 100 刷题计划】 LeetCode 136. 只出现一次的数字 | C++ 哈希表&异或基础解法
c++·算法·leetcode
汉克老师1 小时前
GESP2024年6月认证C++三级( 第二部分判断题(1-10))
c++·数组·位运算·补码·gesp三级·gesp3级
无限进步_2 小时前
【C++】只出现一次的数字 II:位运算的三种解法深度解析
数据结构·c++·ide·windows·git·算法·leetcode
小贾要学习2 小时前
【Linux】TCP网络通信编程
linux·服务器·网络·c++·网络协议·tcp/ip
哎嗨人生公众号3 小时前
手写求导公式,让轨迹优化性能飞升,150ms变成9ms
开发语言·c++·算法·机器人·自动驾驶
code_whiter3 小时前
C++6(模板)
开发语言·c++
一只旭宝3 小时前
【C++ 入门精讲1】初始化、const、引用、内联函数 | 超详细手写笔记(附完整代码)
开发语言·c++
旖-旎3 小时前
哈希表(字母异位次分组)(5)
数据结构·c++·算法·leetcode·哈希算法·散列表
无限进步_4 小时前
【C++】多重继承中的虚表布局分析:D类对象为何有两个虚表?
开发语言·c++·ide·windows·git·算法·visual studio