对象销毁_【C++】对象移动与转发moveforward

点击上方蓝字关注我,我们一起学编程欢迎小伙伴们分享、转载、私信、赞赏

对象移动与转发

1. 移动1.1 右值引用1.2 标准库 move 函数1.3 理解 std::move2. 转发2.1 使用 std::forward 保持类信息

1. 移动

1.1 右值引用

C++11 新标准的一个很重要的特性是可以移动而非拷贝对象的能力。在很多情况下,程序中会发生对象拷贝,但在一些情况下,对象拷贝后就立即被销毁了。在这种情况下,移动而非拷贝对象会有大幅度性能提升。

标准库容器、stringshared_ptr 既支持移动也支持拷贝。IO 类和 unique_ptr 类可以移动但不能拷贝。

为了支持移动操作,新标准引入了一种新的引用类型——右值引用(rvalue reference)。所谓右值引用,就是必须绑定到右值的引用。我们通过 && 而非 & 来获得右值引用。

如我们将要看到的,右值引用有一个很重要的性质——只能绑定到一个将要销毁的对象。因此,我们可以自由地将一个右值引用的资源“移动”到另一个对象中。

一般而言,一个左值表达式表示的是一个对象的身份,而一个右值表达式表示的是对象的值。

类似任何引用,一个右值引用也不过是某个对象的另一个名字而已。如我们所知,对于左值引用,我们不能将其绑定到要求转换的表达式、字面常量或是返回右值的表达式。右值引用有着完全相反的绑定特性:我们可以将一个右值应用绑定到这类表达式上,但不能绑定到一个左值上。

int i = 1;
int &r1 = i;              // 正确:绑定到左值上
int &&rr1 = i;            // 错误:不能绑定到左值上
int &r2 = i * 2;          // 错误:不能绑定到右值上
int &&rr2 = i * 2;        // 正确:绑定到右值表达式
const int &r3 = i * 2;    // 正确:可以将const的引用绑定到右值上

返回左值引用的函数,连同赋值、下标运算、解引用和前置递增/减运算符,都是返回左值的表达式的例子。我们可以将一个左值引用绑定到这类表达式的结果上。

返回非引用类型的函数,连同算术、关系、位以及后置递增/减运算符,都生成右值。我们不能将一个左值引用绑定到这类表达式上,但我们可以将一个 const 的左值引用或者一个右值引用绑定到这类表达式上。

左值持久,右值短暂

考察左值和右值表达式的列表,两者相互区别之处在于:左值有持久的状态;右值要么为字面常量,要么为表达式求值过程中创建的临时对象。

由于右值只能绑定到临时对象,则:(1) 所引用的对象将要销毁;(2) 该对象没有其他用户。这两个特性意味着,使用右值引用的代码可以自由地接管所引用的对象的资源。

1.2 标准库 move 函数

虽然不能将一个右值引用直接绑定到一个左值上,但我们可以显式地将一个左值转换为对应的右值引用类型。通过一个名为 move 的新标准函数来获得绑定到左值上的右值引用,此函数定义在头文件 utility 中。

int i = 1;
int &&rr1 = i;               // 错误
int &&rr2 = std::move(i);    // 正确

move 调用告诉编译器:我们有一个左值,但我们希望像一个右值一样处理它。我们必须认识到,调用 move 就意味着承诺:除了对被移动对象进行销毁和赋值操作外,我们将不再使用它。在调用 move 之后,我们不能对被移动对象的值做任何假设。

我们可以销毁一个被移动对象,也可以对其赋予它新值,但不能使用一个被移动对象的值。

1.3 理解 std::move

由于 move 本质上可以接受任何类型的形参,因此它很顶是一个函数模板。

先来看一下标准库是如何定义 move 函数的:

template<typename T>
typename remove_reference::type&& move(T&& t){return static_cast<typename remove_reference::type&&>(t);
}

首先,move 的函数参数 T&& 是一个指向模板类型参数的引用。通过引用折叠,此参数可以与任何类型的实参匹配(左值、右值)。

引用折叠:所有的引用折叠最终都代表一个引用,要么是左值引用,要么是右值引用。规则就是:如果任一引用为左值引用,则结果为左值引用。否则(即两个都是右值引用),结果为右值引用。(详细解释点我)https://zhuanlan.zhihu.com/p/50816420

再来看一下 move 是如何工作的。我们用两个例子来介绍:

int i = 1;
int j = std::move(i);

函数执行过程:

  • 推断出 T 的类型是 int

  • 因此,remove_reference 用 int 进行实例化

  • remove_reference 的 type 成员是 int

  • move 的返回类型是 int&&

  • move 的函数参数类型是 int&&

int j = std::move(1);

函数执行过程:

  • 推断出 T 的类型是 int& (若为 int ,则最后便是 int&& ,错误)

  • 因此,remove_reference 用 int& 进行实例化

  • remove_reference 的 type 成员是 int

  • move 的返回类型是 int&&

  • move 的函数参数类型是 int& && ,会折叠为 int&

总而言之,对于 std::move 而言,无论传给它的是左值还是右值,通过之后都变成了右值。

2. 转发

某些函数需要将其一个或多个实参连同类型不变地转发给其他函数。在此情况下,我们需要保持被转发实参的所有性质,包括实参类型、是否是 const 、是左值还是右值 。

与前面介绍的相同,通过将一个函数参数定义成一个指向模板类型参数的右值引用,我们可以保持其对应实参的所有类型信息

2.1 使用 std::forward 保持类信息

forward 的作用:

std::forward 被称为完美转发,它的作用是保持原来的属性不变。也就是说,如果原来的值是左值,经 std::forward 处理后该值还是左值;如果原来的值是右值,经 std::forward 处理后还是右值。

下面我们来看一个例子:

#include 

template<typename T>
void print(T& t){
    std::cout <"l-value" <std::endl;
}

template<typename T>
void print(T&& t){
    std::cout <"r-value" <std::endl;
}

template<typename T>
void TestForward(T&& v){
    print(v);
    print(std::move(v));
    print(std::forward(v));
}int main(){int x = 1;
    TestForward(x);std::cout <"*********" <std::endl;
    TestForward(1);return 0;
}/*
编译运行:
jincheng@haofan$ g++ test.cpp
jincheng@haofan$ ./a.out
l-value
r-value
l-value
*********
l-value
r-value
r-value
jincheng@haofan$
*/

下面我们分析一下:

在第一组中,输入的参数是“左值 x” 。因此,第一行输出左值;第二行输出右值;第三行保持 x 的左值特性,所以输出左值。
在第二组中,输入的参数是“右值 1” 。第一行由于形参 x 分配了内存空间,已经变为左值了,所以输出左值;第二行输出右值;第三行保持 x 的右值特性,所以输出右值。

325fc60958dc718b3eb76bafb03520c2.png

长按识别关注

*编程笔记本*

来都来了,点个在看再走吧~~~

147c87ed7057a4d3ae9f25e7ef556a0b.gif
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值