C++ 右值语义相关总结

在现代C++的众多特性中,右值语义(std::move和std::forward)大概是最神奇也最难懂的特性之一了。本文简要介绍了现代C++中右值语义特性的原理和使用。

1 什么是左值,什么是右值?

int a = 0;  // a是左值,0是右值
int b = rand(); // b是左值,rand()是右值

直观理解:左值在等号左边,右值在等号右边

深入理解:左值有名称,可根据左值获取其内存地址,而右值没有名称,不能根据右值获取地址。

2 引用叠加规则

左值引用A&和右值引用A&&可相互叠加, 叠加规则如下:

A& + A& = A&
A& + A&& = A&
A&& + A& = A&
A&& + A&& = A&&

举例说明,在模板函数void foo(T&& x)中:

如果T是int&类型, T&&为int&,x为左值语义
如果T是int&&类型, T&&为int&&, x为右值语义
也就是说,不管输入参数x为左值还是右值,都能传入函数foo。区别在于两种情况下,编译器推导出模板参数T的类型不一样。

3 std::move

3.1 What?

在C++11中引入了std::move函数,用于实现移动语义。它用于将临时变量(也有可能是左值)的内容直接移动给被赋值的左值对象。

3.2 Why?

知道了std::move是干什么的,他能给我们的搬砖工作带来哪些好处呢? 举例说明:

如果类X包含一个指向某资源的指针,在左值语义下,类X的复制构造函数定义如下:

X::X()
{
 // 申请资源(指针表示)
}

X::X(const X& other)
{
 // ...
 // 销毁资源
 // 克隆other中的资源
 // ...
}

X::~X()
{
 // 销毁资源
}

假设应用代码如下。其中,对象tmp被赋给a之后,便不再使用。

X tmp;
// ...经过一系列初始化...
X a = tmp;

在上面的代码中,执行步骤:

  • 先执行一次默认构造函数(默认构造tmp对象)
  • 再执行一次复制构造函数(复制构造a对象)
  • 退出作用域时执行析构函数(析构tmp和a对象)

从资源的视角来看,上述代码中共执行了2次资源申请和3次资源释放。

那么问题来了,既然对象tmp只是一个临时对象,在执行X a = tmp;时,对象a能否将tmp的资源'偷'过来,直接为我所用,而不影响原来的功能? 答案是可以。

X::X(const X& other)
{
 // 使用std::swap交换this和other的资源
}

通过'偷'对象tmp的资源,减少了资源申请和释放的开销。而std::swap交换指针代价极小,可忽略不计。

3.3 How?

到现在为止,我们明白了std::move将要达到的效果,那么它究竟是怎么实现的呢?

template<class T>
typename remove_reference<T>::type&&
std::move(T&& a) noexcept
{
 typedef typename remove_reference<T>::type&& RvalRef;
 return static_cast<RvalRef>(a);
}

不管输入参数为左值还是右值,都被remove_reference去掉其引用属性,RvalRef为右值类型,最终返回类型为右值引用。

3.4 Example

在实际使用中,一般将临时变量作为std::move的输入参数,并将返回值传入接受右值类型的函数中,方便其'偷取'临时变量中的资源。需要注意的是,临时变量被'偷'了之后,便不能对其进行读写,否则会产生未定义行为。

#include <utility>
#include <iostream>
#include <string>
#include <vector>    

void foo(const std::string& n)
{
 std::cout << "lvalue" << std::endl;
}        

void foo(std::string&& n)
{
 std::cout << "rvalue" << std::endl;
}        

void bar()
{
 foo("hello");    // rvalue
 std::string a = "world";
 foo(a);      // lvalue
 foo(std::move(a));   // rvalue
}

int main()
{
 std::vector<std::string> a = {"hello", "world"};
 std::vector<std::string> b;

 b.push_back("hello");   // 开销:string复制构造
 b.push_back(std::move(a[1])); // 开销:string移动构造(将临时变量a[1]中的指针偷过来)

 std::cout << "bsize: " << b.size() << std::endl;
 for (std::string& x: b)
 std::cout << x << std::endl;
 bar();
 return 0;
}

4 std::forward

4.1 What?

std::forward用于实现完美转发。那么什么是完美转发呢?完美转发实现了参数在传递过程中保持其值属性的功能,即若是左值,则传递之后仍然是左值,若是右值,则传递之后仍然是右值。

简单来说,std::move用于将左值或右值对象强转成右值语义,而std::forward用于保持左值对象的左值语义和右值对象的右值语义。

4.2 Why?

#include <utility>
#include <iostream>

void bar(const int& x)
{
 std::cout << "lvalue" << std::endl;
}

void bar(int&& x)
{
 std::cout << "rvalue" << std::endl;
}

template <typename T>
void foo(T&& x)
{
 bar(x);
}

int main()
{
 int x = 10;
 foo(x); // 输出:lvalue
 foo(10); // 输出:lvalue
 return 0;
}

执行以上代码会发现,foo(x)和foo(10)都会输出lvalue。foo(x)输出lvalue可以理解,因为x是左值嘛,但是10是右值,为啥foo(10)也输出lvalue呢?

这是因为10只是作为函数foo的右值参数,但是在foo内部,10被带入了形参x,而x是一个有名字的变量,即右值,因此foo中bar(x)还是输出lvalue。

那么问题来了,如果我们想在foo函数内部保持x的右值语义,该怎么做呢?std::forward便派上了用场。

只需改写foo函数:

template <typename T>
void foo(T&& x)
{
 bar(std::forward<T>(x));
}

4.3 How?

std::forward听起来有点神奇,那么它到底是如何实现的呢?

template<typename T, typename Arg>
shared_ptr<T> factory(Arg&& arg)
{
 return shared_ptr<T>(new T(std::forward<Arg>(arg)));
}
template<class S>
S&& forward(typename remove_reference<S>::type& a) noexcept
{
 return static_cast<S&&>(a);
}

X x;
factory<A>(x);

如果factory的输入参数是一个左值,那么Arg = X&,根据叠加规则,std::forward<Arg> = X&。因此,在这种情况下,std::forward<Arg>(arg)仍然是左值。

相反,如果factory输入参数是一个右值,那么Arg = X,std::forward<Arg> = X。这种情况下,std::forward<Arg>(arg)是一个右值。

恰好达到了保留左值or右值语义的效果!

4.4 Example

直接上代码。如果前面都懂了,相信这段代码的输出结果也能猜个八九不离十了。

#include <utility>
#include <iostream>

void overloaded(const int& x)
{
 std::cout << "[lvalue]" << std::endl;
}

void overloaded(int&& x)
{
 std::cout << "[rvalue]" << std::endl;
}

template <class T> void fn(T&& x)
{
 overloaded(x);
 overloaded(std::forward<T>(x));
}

int main()
{
 int i = 10;
 overloaded(std::forward<int>(i));
 overloaded(std::forward<int&>(i));
 overloaded(std::forward<int&&>(i));

 fn(i);
 fn(std::move(i));

 return 0;
}

以上就是C++ 右值语义相关总结的详细内容,更多关于C++ 右值语义的资料请关注我们其它相关文章!

(0)

相关推荐

  • 浅析C++11中的右值引用、转移语义和完美转发

    1. 左值与右值: C++对于左值和右值没有标准定义,但是有一个被广泛认同的说法:可以取地址的,有名字的,非临时的就是左值;不能取地址的,没有名字的,临时的就是右值. 可见立即数,函数返回的值等都是右值;而非匿名对象(包括变量),函数返回的引用,const对象等都是左值. 从本质上理解,创建和销毁由编译器幕后控制的,程序员只能确保在本行代码有效的,就是右值(包括立即数);而用户创建的,通过作用域规则可知其生存期的,就是左值(包括函数返回的局部变量的引用以及const对象),例如: int& fo

  • 深入了解c++11 移动语义与右值引用

    1.移动语义 C++11新标准中一个最主要的特性就是提供了移动而非拷贝对象的能力.如此做的好处就是,在某些情况下,对象拷贝后就立即被销毁了,此时如果移动而非拷贝对象会大幅提升性能.参考如下程序: //moveobj.cpp #include <iostream> #include <vector> using namespace std; class Obj { public: Obj(){cout <<"create obj" << e

  • 详解C++11中的右值引用与移动语义

    C++11的一个最主要的特性就是可以移动而非拷贝对象的能力.很多情况都会发生对象的拷贝,有时对象拷贝后就立即销毁,在这些情况下,移动而非拷贝对象会大幅度提升性能. 右值与右值引用 为了支持移动操作,新标准引入了一种新的引用类型--右值引用,就是必须绑定到右值的引用.我们通过&&而不是&来获得右值引用.右值引用一个重要的特性就是只能绑定到将要销毁的对象. 左值和右值是表达式的属性,一些表达式生成或要求左值,而另一些则生成或要求右值.一般而言,一个左值表达式表示的是一个对象的身份,而右

  • C++ 右值语义相关总结

    在现代C++的众多特性中,右值语义(std::move和std::forward)大概是最神奇也最难懂的特性之一了.本文简要介绍了现代C++中右值语义特性的原理和使用. 1 什么是左值,什么是右值? int a = 0; // a是左值,0是右值 int b = rand(); // b是左值,rand()是右值 直观理解:左值在等号左边,右值在等号右边 深入理解:左值有名称,可根据左值获取其内存地址,而右值没有名称,不能根据右值获取地址. 2 引用叠加规则 左值引用A&和右值引用A&&a

  • C++11右值引用和转发型引用教程详解

    右值引用 为了解决移动语义及完美转发问题,C++11标准引入了右值引用(rvalue reference)这一重要的新概念.右值引用采用T&&这一语法形式,比传统的引用T&(如今被称作左值引用 lvalue reference)多一个&. 如果把经由T&&这一语法形式所产生的引用类型都叫做右值引用,那么这种广义的右值引用又可分为以下三种类型: 无名右值引用 具名右值引用 转发型引用 无名右值引用和具名右值引用的引入主要是为了解决移动语义问题. 转发型引用的引

  • C++右值引用与move和forward函数的使用详解

    目录 1.右值 1.1 简介 1.2 右值引用 1.3 右值引用的意义 2.move 3.foward 1.右值 1.1 简介 首先区分一下左右值: 左值是指存储在内存中.有明确存储地址(可取地址)的数据: 右值是指可以提供数据值的数据(不可取地址) 如int a=123:123是右值, a是左值.总的来说 可以对表达式取地址(&)就是左值,否则为右值 而C++11 中右值又可以分为两种: 纯右值:非引用返回的临时变量.运算表达式产生的临时变量如a+b.原始字面量和 lambda 表达式等 将亡

  • 详解C++中的左值,纯右值和将亡值

    目录 引入 一.表达式 二.值类别 三.左值 四.纯右值 五.将亡值 六.注意 引入 C++中本身是存在左值,右值的概念,但是在C11中又出现了左值,纯右值,将亡值得概念:这里我们主要介绍这些值的概念. 一.表达式 定义:由运算符和运算对象构成的计算式(类似数学中的算术表达式) 每个 C++ 表达式(带有操作数的操作符.字面量.变量名等)可按照两种独立的特性加以辨别:**类型和值类别 **(value category).每个表达式都具有某种非引用类型,且每个表达式只属于三种基本值类别中的一种:

  • 深入学习C++智能指针之shared_ptr与右值引用的方法

    目录 1. 介绍 2. 初始化方法 2.1 通过构造函数初始化 2.2 通过拷贝和移动构造函数初始化 2.3 通过 std::make_shared 初始化 2.4 通过 reset 方法初始化 3. 获取原始指针 4. 指定删除器 5. 参考链接 1. 介绍 在 C++ 中没有垃圾回收机制,必须自己释放分配的内存,否则就会造成内存泄露.解决这个问题最有效的方法是使用智能指针(smart pointer).智能指针是存储指向动态分配(堆)对象指针的类,用于生存期的控制,能够确保在离开指针所在作用

  • C++11右值引用和移动语义的实例解析

    目录 基本概念 左值 vs 右值 左值引用 vs 右值引用 右值引用使用场景和意义 左值引用的使用场景 左值引用的短板 右值引用和移动语义 右值引用引用左值 右值引用的其他使用场景 完美转发 万能引用 完美转发保持值的属性 完美转发的使用场景 总结 基本概念 左值 vs 右值 什么是左值? 左值是一个表示数据的表达式,如变量名或解引用的指针. 左值可以被取地址,也可以被修改(const修饰的左值除外). 左值可以出现在赋值符号的左边,也可以出现在赋值符号的右边. int main() { //以

  • C++左值与右值,右值引用,移动语义与完美转发详解

    目录 C++——左值与右值.右值引用.移动语义与完美转发 一.左值和右值的定义 二.如何判断一个表达式是左值还是右值(大多数场景) 三.C++右值引用 四.std::move()与移动语义 五. 完美转发 总结 C++——左值与右值.右值引用.移动语义与完美转发 在C++或者C语言中,一个表达式(可以是字面量.变量.对象.函数的返回值等)根据其使用场景不同,分为左值表达式和右值表达式. 一.左值和右值的定义 1.左值的英文为locator value,简写为lvalue,可意为存储在内存中.有明

随机推荐