C++ std::move()和完美转发

14 篇文章 0 订阅

std::move()、std::forward<T>、模板类型推断分析

 

引用折叠原则和完美转发是有联系的,可以说后者是基于前者的某些特性实现的,具体来看一下。

要理解完美转发,需要了解两个知识点:

  1. 引用折叠原则(Reference collapsing rules)。

  2. 右值函数模版参数类型推导(Template argument deduction)

我们先来分析一下为什么需要使用到move呢?

C++11多出来一个move语义,意图是解决临时对象重复拷贝和释放引发的资源浪费,move与右值引用进行搭配可以完美的解决这个问题。
比如在进行vector中的insert函数的时候,如果模板类型是string,通过move(str1)我们就可以只将string中的指针进行交换,即可实现插入到vector容器中的操作。而不用再进行深拷贝的动作。
需要注意的是,如果使用move进行左值变右值之后,该左值不能再利用,因为它里面的信息可能已经被更改,对应的string就是,指向字符串的指针被打断了。

下面这张图是模板类型推断原则:

 
这是函数模板参数类型推导中一种比较特殊的情况,这种情况会把模板参数作为右值引用使用,例如:

template<typename T>  
void foo(T&&);  
  •  

其中T为模板类型,T&&为参数类型。这种情况会产生两种结果:

1. 当传给foo函数的参数是一个左值引用时,例如:

int i = 29;  
foo(i);//i为左值引用  
  •  

此时,T的类型为int的左值引用:int&,参数类型为int & &&,(既T&&),结合上面的引用折叠规则,最终参数的类型为int的左值引用:int&。
2. 当传给foo函数的参数是一个右值引用时,例如:

foo(29);
  •  

此时,T的类型为int,参数类型为int&&,(既T&&)。

那么,为什么需要forward呢?
我们先来看下以下例子:

template<typename T>
void show1(T && a)
{
	cout << "我是右值引用" << endl;
}

template<typename T>
void show1(T & a)
{
	cout << "我是左值引用" << endl;
}

template<typename T>
void show(T && a)
{
	show1(forward<T>(a));
}
//我们通过以下进行调用;
show(2);

  •  

该函数第进入show之后会调用show1的调用,如果此时没有用forward进行转发的话,会调用到
左值引用版本,而不会调用到右值引用的版本。所有forward就是进行完美转发的作用,当传入的是左值引用
的时候,会调用到左值引用的版本,当调用到右值引用的时候,函数里面也会调用到右值引用的版本 那么他是怎么实现的呢?
通过以上代码的分析,我们可以发现,forward会进行类型的转发。具体实现原理就是上面所说的引用折叠原则。

下面是VS2015下的实现源码。

// TEMPLATE FUNCTION forward
template<class _Ty> inline
	constexpr _Ty&& forward(
		typename remove_reference<_Ty>::type& _Arg) _NOEXCEPT
	{	// forward an lvalue as either an lvalue or an rvalue
	return (static_cast<_Ty&&>(_Arg));
	}

template<class _Ty> inline
	constexpr _Ty&& forward(
		typename remove_reference<_Ty>::type&& _Arg) _NOEXCEPT
	{	// forward an rvalue as an rvalue
	static_assert(!is_lvalue_reference<_Ty>::value, "bad forward call");
	return (static_cast<_Ty&&>(_Arg));
	}

我们就上面那个例子来进行分析:
1、传入的是左值的时候。

void show(int&   && a)
{
	//此时T为 int&类型。a通过引用折叠原则为int&类型
	show1(forward<int&>(a));
}

此时forward会调用下面这个版本的函数。

	constexpr int& && forward(
		typename remove_reference<int&>::type& _Arg) _NOEXCEPT
	{	// forward an lvalue as either an lvalue or an rvalue
	return (static_cast<int& &&>(_Arg));
	}

那么,此时返回的就是int& &&类型,通过引用折叠原则即为 int &,完成了左值的转发

2、传入的是右值的时候。

下面是类型推断调用的时候生成的函数。

void show(int && a)
{
	//此时T为 int类型。a为 int&& 类型
	show1(forward<int>(a));
}
  •  

相应地就会调用到下面这个forward版本:

constexpr int && forward(
	typename remove_reference<int>::type&& _Arg) _NOEXCEPT
{	// forward an rvalue as an rvalue
static_assert(!is_lvalue_reference<int>::value, "bad forward call");
return (static_cast<int &&>(_Arg));
}

我们可以看出上面的这个forward会返回右值引用,则实现了右值引用的完美转发。

总结

1、move函数是将一个左值转为右值引用,转换之后,改左值就不能再进行使用了。
2、forward函数是进行类型的转发,可以将左值引用继续转发为左值,右值引用继续转发为右值引用。

### 回答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::forward是C++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、付费专栏及课程。

余额充值