std::move与std::forward

std::move与std::forward

这俩家伙显然都是c++11的产物无疑了。分别叫做:移动完美转发。如果你写过模板函数,那么你十有八九会碰到std::forward;如果你经常使用智能指针(特别是unique),或者需要用移动来避免深拷、提高效率,那么你一定会碰到std::move。那么,下面我们来分别理解一下他俩啥原理、咋用的吧!

1. std::move——移动

首先需要记一个重要的结论:std::move什么也没做,只是营造一种”移动语义“。

理解上面这句话非常重要,什么叫营造”移动语义“呢?先来看看std::move的源码:

//c++11的实现,c++14中就可以用decltype(auto)啦
template<typename _Tp>
    constexpr typename std::remove_reference<_Tp>::type&&
    move(_Tp&& __t) noexcept
    { return static_cast<typename std::remove_reference<_Tp>::type&&>(__t); }

可见std::move内部唯一做的事情就是把传入的对象强转成一个“右值”返回。但要十分注意的是,std::move返回的”右值“,不是真的右值。即,并不是那种生命周期只有一行代码的那种!std::move没有这种“神奇”的功效。(经过std::move返回后变量在后文就不能用了?并不是这样!)

既然如此,那是不是可以说这个std::move并没有真的移动呢?是的,std::move其实并没有真的移动对象!

那这玩意有啥用啊???它的作用就是,可以帮你触发你写的(或别人写的)类中的m-ctor或m=。比如:

std::vector<int> arr1{1,2,3};
std::vector<int> arr2(std::move(arr1));

这样就触发了std::vector的m-ctor,因为m-ctor接受一个右值引用(T&&),而std::move正好又返回一个T&&。

进入m-ctor或者m=之后呢,按照规范来说 为了保证移动操作的效率,**必须要对类中的大资源进行浅拷贝,然后销毁被移动对象。**这就是m-ctor、m=必须要做的两件事。也就是说,std::move并不会真的移动对象,而真正起到移动作用的是m-ctor和m=!

可以用上面的代码简单验证一下std::vector的移动行为:

//继续上面的代码
std::cout<<arr2.size()<<std::endl;//输出3
arr1[1]=2;//只有这一句,Segmentation fault
arr1.push_back(1);//只有这一句,没事

可见,std::vector中的移动操作是这样的:把被移动数组清空但不销毁。你可以接着用arr1,但它已经被清空了。但这个操作一定是std::vector的m-ctor或m=做的,一定不是std::move做的!这一点不要混淆:

std::vector<int> arr1{1,2,3};
std::move(arr1);
arr1[1]=2;//没事,没段错误!
std::cout<<arr1.size()<<std::endl;//输出3

另外,《mordern c++》里关于m-ctor和m=提到一点建议:m-ctor和m=最好标记为noexcept,可以尽可能提高标准库容器的移动效率。具体原因在以前的博客中讲过了。【链接】

所以,现在知道应该怎么写你自己的m-ctor和m=了吧!(答案都在上面了)

另外,对于智能指针而言,使用std::move意味着被移动的指针失去了其对象的所有权,不论是unique还是shared。一般对unique使用std::move的场景常见。

2. std::forward——完美转发

我觉得,要理解完美转发的原理和用途,比std::move略复杂,需要先搞清楚模板推导与万能引用的概念【链接】。这里我们再复习一下:

template<typename T>
void fun(T&& arg){
    //假设函数process有2个版本的重载,分别接受左值和右值
    process(std::forward<T>(arg));
}
//调用处:
int a=0;
fun(a);//T->int&,arg->int& && -> 引用折叠 -> int&
fun(2);//T->int,arg->int&&

简单来说,万能引用的作用在于,可以根据实参的左值右值类型推导出形参是左值引用还是右值引用。但在我们编写fun模板函数时,经常会碰到这样的问题:不论arg形参被推导成左值引用or右值引用,arg一定是一个左值;如果我现在想让arg形参还原成实参真实的左右值性质该怎么办?(即 在编写函数时有什么办法能得知传入的实参到底是左值还是右值?)

这就需要std::forward的帮助了!

std::forward相对std::move而言是“有条件”的转发,并不是像std::move那样把输入参数一股脑地转发成右值返回。而是这样做的:当输入的模板参数T是被推导成左值引用时,将传入实参转发为左值返回;当输入的模板参数T被推到成右值时,将传入的实参转发为右值返回。

这也是为什么std::forward是个模板函数,而std::move不需要模板参数。看上面的代码注释处,万能引用arg在传入时是左值or右值,关键看模板类型T即可区分!

到此,你应该明白移动和完美转发的本质原理以及用途,其实二者本质上都是cast而已,为了写法方便才有了标准库的函数。理解了本质,自然就知道此二者什么场合用、该怎么用了。

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: c++中的std::move和std::forward都是用于实现完美转发的工具。 std::move是将一个左值强制转换为右值引用,从而实现将资源所有权从一个对象转移到另一个对象的目的。使用std::move可以避免不必要的复制和赋值操作,提高程序的效率。 std::forward则是用于在函数模板中实现完美转发,将参数按照原来的类型转发给下一个函数。它可以保证参数的类型和值被完美地转发,避免了不必要的拷贝和移动操作,提高了程序的效率。 总的来说,std::move和std::forward都是用于提高程序效率和避免不必要的拷贝和移动操作的工具。 ### 回答2: C++标准库中提供了两个模板函数std::move和std::forward,它们在C++11中引入,用于实现移动语义和完美转发。 std::move的作用是将一个左值强制转换为右值引用,使得该对象的所有权能够被转移,而不是进行复制或者赋值。通过调用移动构造函数或者移动赋值运算符来减少开销。移动语义是C++11中的一个重要特性,它可以提高程序的效率并且使得程序更加高效。 std::forward的作用是实现完美转发,将函数参数原封不动地转发到另一个函数中,使得函数模板可以保持参数类型和实参类型一致。std::forward用于实现通用类型的泛型编程,解决了模板函数中参数类型无法确定的问题。 实际上,std::move和std::forward的实现方式都非常简单,都是使用了static_cast进行类型转换。但是它们在C++11中的引入,以及其实现的本质却给C++程序的效率提高和泛型编程提供了重要的支持。 总之,std::move和std::forwardC++11中非常重要的语言特性,它们可以帮助程序员实现移动语义和完美转发,提高程序的性能和可读性。要注意正确使用它们,以避免出现不必要的开销和错误。 ### 回答3: C++ 11中引入了两个新的特殊函数模板std::move()和std::forward(),用来实现完美转发和移动语义,提高了代码的效率和简洁性。 std::move的作用就是将一个左值转换成右值引用,将左值的所有权抢过来,但不进行任何内存拷贝。通常用于移动语义,可以提高程序的效率。用法很简单,就是std::move(左值变量)。比如,若有个vector<int> a和一个vector<int> b,我想把b中的元素全部移动到a中,可以这样写:a.insert(a.end(), std::make_move_iterator(b.begin()), std::make_move_iterator(b.end()));这里,std::make_move_iterator()是一个语法糖,将它们的元素包装成可以引用的右值。 std::forward的作用是保持参数本来的类型(左值或右值),既可以接收左值也可以接收右值,并将参数传递给其他函数,这就是所谓的完美转发。完美转发可以达到只有一个函数就可以处理所有情况的目的。用法就是std::forward<参数类型>(参数变量)。比如,若有个函数template<class T> void f(T&& t),其中参数t是万能引用,需要把t传递给其他函数g(),我们可以这样写:g(std::forward<T>(t));这样就可以达到完美转发的目的。 需要注意的是,std::move和std::forward虽然看起来相似,但作用是不同的,std::move是将左值转换成右值引用,而std::forward是维持参数的原类型,用于完美转发。同时,它们都需要加上相应的模板类型,以便让编译器进行类型推导。在使用时,需要根据情况选择合适的函数,以达到更好的效果。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值