左值(Lvalue)与右值(Rvalue)
英文含义:
-
左值(Lvalue):Locator value,意味着它指向一个具体的内存位置。
-
右值(Rvalue):Read value,指的是可以读取的数据,但不一定指向一个固定的内存位置。
定义
- 左值:指的是一个持久的内存地址。左值可以出现在赋值操作的左侧或右侧。例如,变量、数组的元素、对对象成员的引用等都是左值。
- 右值:通常是临时的、不能有多个引用的值,它们不指向持久的内存地址。右值可以出现在赋值操作的右侧,但不能出现在左侧。字面量(如42、3.14)、临时对象、以及返回临时对象的表达式等都是右值。
完美转发(Perfect Forwarding)
完美转发是C++11引入的一个概念,其目的是允许函数模板将参数以原来的左值或右值的形式转发到其他函数。这是通过引用折叠规则和std::forward
函数实现的。完美转发的一个关键应用场景是模板函数中,我们希望将接收到的参数以完全相同的形式(保持其左值或右值性质)传递给另一个函数时使用。
引用折叠规则
在模板函数或类中,当一个引用的引用被形成时,它们会折叠成单一的引用
T& &
,T& &&
,T&& &
都会被折叠为T&
T&& &&
会被折叠为T&&
示例
- 当
wrapper(lv)
被调用时,lv
是一个左值,因此模板参数T
被推断为int&
(左值引用)。由于引用折叠规则,T&&
折叠为int&
。因此,std::forward<T>(arg)
将arg
作为左值引用转发给process
函数,调用process(int& i)
版本。 - 当
wrapper(20)
被调用时,20
是一个右值,因此模板参数T
被推断为int
。由于T
是一个非引用类型,T&&
就直接是int&&
(右值引用)。因此,std::forward<T>(arg)
将arg
作为右值引用转发给process
函数,调用process(int&& i)
版本。
highlighter- arduino
#include <iostream> #include <utility> // std::forward // 分别处理左值和右值 void process(int& i) { std::cout << "Process left value: " << i << std::endl; } void process(int&& i) { std::cout << "Process right value: " << i << std::endl; } // 完美转发的模板函数 template<typename T> void wrapper(T&& arg) { // 使用std::forward来完美转发arg process(std::forward<T>(arg)); } int main() { int lv = 10; // 左值 wrapper(lv); // arg被推断为左值引用,因为lv是一个左值 wrapper(20); // 20是右值,arg被推断为右值引用 return 0; } /* Process left value: 10 Process right value: 20 */
转移(Move)
转移是指将一个对象的资源(如动态内存)从一个实例转移到另一个实例,而不是复制资源。这通常通过移动构造函数和移动赋值操作符实现,它们接受一个右值引用(Rvalue reference)作为参数。移动语义允许资源的高效转移,避免了不必要的复制,特别是对于大型对象或资源密集型对象。
使用
std::move
方法可以将左值转换为右值。使用这个函数并不能移动任何东西,而是和移动构造函数一样都具有移动语义,将对象的状态或者所有权从一个对象转移到另一个对象,只是转移,没有内存拷贝。
从实现上讲,
std::move
基本等同于一个类型转换:static_cast<T&&>(lvalue);
,函数原型如下:
highlighter- cpp
template<class _Ty> _NODISCARD constexpr remove_reference_t<_Ty>&& move(_Ty&& _Arg) _NOEXCEPT { // forward _Arg as movable return (static_cast<remove_reference_t<_Ty>&&>(_Arg)); }
联系
- 完美转发和移动语义都紧密依赖于左值和右值的概念。完美转发用于保持参数的左值或右值性质不变,而移动语义则是利用右值(通常是即将销毁的临时对象)来优化资源的管理。
- 移动语义是完美转发常见的一个应用场景。当使用完美转发将函数参数传递给另一个函数时,如果该参数是一个临时对象(右值),则可以利用移动构造函数或移动赋值操作符,从而提高效率。
- 右值的概念是转移语义的基础。只有右值(临时对象或显式标记为右值的对象)才能被移动,以此来优化资源的使用和提高程序的运行效率。
示例
以下是一个简单示例,其中包含一个自定义的String
类,这个类通过实现移动构造函数和移动赋值操作符来优化内存资源管理。
同时,代码使用完美转发的函数模板,它可以根据传入参数的类型(左值或右值)来决定是否使用移动语义。
https://www.bilibili.com/read/cv32911818/
https://www.bilibili.com/read/cv32909725/
https://www.bilibili.com/read/cv32910298/
https://www.bilibili.com/read/cv32915294/
https://www.bilibili.com/read/cv32915343/
https://www.bilibili.com/read/cv32915382/
https://www.bilibili.com/read/cv32915437/
https://www.bilibili.com/read/cv32915539/
https://www.bilibili.com/read/cv32915608/
https://www.bilibili.com/read/cv32915669/
https://www.bilibili.com/read/cv32915709/
https://www.bilibili.com/read/cv32915754/
https://www.bilibili.com/read/cv32915799/
https://www.bilibili.com/read/cv32915836/
https://www.bilibili.com/read/cv32915868/
https://www.bilibili.com/read/cv32915947/
https://www.bilibili.com/read/cv32915997/
https://www.bilibili.com/read/cv32916031/
https://www.bilibili.com/read/cv32916905/
https://www.bilibili.com/read/cv32916958/
https://www.bilibili.com/read/cv32917039/
https://www.bilibili.com/read/cv32917159/
https://www.bilibili.com/read/cv32917293/
https://www.bilibili.com/read/cv32917398/
https://www.bilibili.com/read/cv32917505/
https://www.bilibili.com/read/cv32917576/
https://www.bilibili.com/read/cv32917611/
https://www.bilibili.com/read/cv32917758/
https://www.bilibili.com/read/cv32917808/
https://www.bilibili.com/read/cv32911944/
https://www.bilibili.com/read/cv32912015/
https://www.bilibili.com/read/cv32912052/
https://www.bilibili.com/read/cv32912077/
https://www.bilibili.com/read/cv32912177/
https://www.bilibili.com/read/cv32912219/
https://www.bilibili.com/read/cv32912253/
https://www.bilibili.com/read/cv32912326/
https://www.bilibili.com/read/cv32914739/
https://www.bilibili.com/read/cv32914734/
https://www.bilibili.com/read/cv32914727/
https://www.bilibili.com/read/cv32914721/
https://www.bilibili.com/read/cv32914713/
https://www.bilibili.com/read/cv32914704/
https://www.bilibili.com/read/cv32914701/
https://www.bilibili.com/read/cv32914693/
https://www.bilibili.com/read/cv32914689/
https://www.bilibili.com/read/cv32914686/
https://www.bilibili.com/read/cv32914684/
https://www.bilibili.com/read/cv32913919/
https://www.bilibili.com/read/cv32913885/
https://www.bilibili.com/read/cv32913866/
https://www.bilibili.com/read/cv32913831/
https://www.bilibili.com/read/cv32913810/
https://www.bilibili.com/read/cv32913780/
https://www.bilibili.com/read/cv32913743/
https://www.bilibili.com/read/cv32913708/
https://www.bilibili.com/read/cv32913653/
https://www.bilibili.com/read/cv32913618/
https://www.bilibili.com/read/cv32913596/
https://www.bilibili.com/read/cv32913583/
https://www.bilibili.com/read/cv32913565/
https://www.bilibili.com/read/cv32913545/
https://www.bilibili.com/read/cv32913526/
https://www.bilibili.com/read/cv32913503/
https://www.bilibili.com/read/cv32913463/
https://www.bilibili.com/read/cv32913435/
https://www.bilibili.com/read/cv32913420/
https://www.bilibili.com/read/cv32913400/
https://www.bilibili.com/read/cv32913390/
https://www.bilibili.com/read/cv32913371/
https://www.bilibili.com/read/cv32913350/
https://www.bilibili.com/read/cv32913325/
https://www.bilibili.com/read/cv32913301/
https://www.bilibili.com/read/cv32913270/
https://www.bilibili.com/read/cv32913241/
https://www.bilibili.com/read/cv32913194/
https://www.bilibili.com/read/cv32913165/
https://www.bilibili.com/read/cv32913147/
https://www.bilibili.com/read/cv32913129/
https://www.bilibili.com/read/cv32913118/
https://www.bilibili.com/read/cv32913097/
https://www.bilibili.com/read/cv32913083/
https://www.bilibili.com/read/cv32913058/
https://www.bilibili.com/read/cv32912741/
https://www.bilibili.com/read/cv32912718/
https://www.bilibili.com/read/cv32912694/
https://www.bilibili.com/read/cv32912527/
https://www.bilibili.com/read/cv32916868/
https://www.bilibili.com/read/cv32917248/
https://www.bilibili.com/read/cv32917859/
https://www.bilibili.com/read/cv32918364/
https://www.bilibili.com/read/cv32918180/
https://www.bilibili.com/read/cv32918205/
https://www.bilibili.com/read/cv32918220/
https://www.bilibili.com/read/cv32918254/
https://www.bilibili.com/read/cv32918382/
https://www.bilibili.com/read/cv32918422/
https://www.bilibili.com/read/cv32918454/
https://www.bilibili.com/read/cv32918499/
https://www.bilibili.com/read/cv32918777/
https://www.bilibili.com/read/cv32918805/