我有一个在矢量上运行的代码:
template<typename T>
void doVector(vector<T>& v, T&& value) {
//....
v.push_back(value);
//...
}
对于普通push_back
,我是否需要使用forward(value)
,move(value)
或value
(根据新的C ++ 11)?它们如何影响性能呢?
例如,
v.push_back(forward<T>(value));
答案 0 :(得分:16)
当第二个参数为 lvalue 时,当前代码将无法编译,因为T&&
将变为X&
,这意味着T
需要X&
1}}这反过来意味着std::vector<T>
将成为std::vector<X&>
,它将与std::vector<X> &
的第一个参数不匹配。因此,这是一个错误。
我会使用两个模板参数:
template<typename T, typename V>
void doVector(vector<T> & v, V && value)
{
v.emplace_back(std::forward<V>(value));
}
由于V
可能与T
的类型不同,因此emplace_back
会使更多感觉,因为它不仅解决了问题,而且使代码更多通用的。 : - )
现在是下一个改进:因为我们正在使用emplace_back
从参数T
创建类型value
的对象(可能使用构造函数),我们可以利用这个事实,并使其具有可变功能:
template<typename T, typename ...V>
void doVector(vector<T> & v, V && ... value)
{
v.emplace_back(std::forward<V>(value)...);
}
这更通用,因为您可以将其用作:
struct point
{
point(int, int) {}
};
std::vector<point> pts;
doVector(pts, 1, 2);
std::vector<int> ints;
doVector(ints, 10);
希望有所帮助。
答案 1 :(得分:6)
forward(value)
,保留l值,r值等内容。
转发非常有用,因为它可以帮助您避免为具有l-val,r-val和引用参数的不同组合的函数编写多个重载
move(value)
实际上是一种将l值转换为r值的转换运算符
在表演方面,都避免制作额外的物品副本,这是主要的好处。
所以他们真的做了两件不同的事情
当你说普通的push_back时,我不确定你的意思,这里有两个签名。
void push_back( const T& value );
void push_back( T&& value );
第一个你可以通过任何正常的l-val,但是对于第二个你必须“移动”一个l-val或向前一个r-val。请记住,一旦移动l-val就无法使用它
这里的奖励是resource,似乎可以很好地解释r-val-refs的概念以及与之相关的其他概念。
正如其他人所建议你也可以切换到使用emplace,因为它实际上完美地将参数转发给对象的构造函数,这意味着你可以随意传递它。
答案 2 :(得分:6)
std::forward
。std::move
。E.g。
template <typename T>
void funcA(T&& t) {
funcC(std::forward<T>(t)); // T is deduced and therefore T&& means forward-ref.
}
void funcB(Foo&& f) {
funcC(std::move(f)); // f is r-value, use move.
}
以下是Scott Meyers的精彩视频,解释转发参考(他称之为通用参考)以及何时使用完美转发和/或std::move
:
C++ and Beyond 2012: Scott Meyers - Universal References in C++11
有关使用std::forward
:Advantages of using forward
答案 3 :(得分:0)
该视频IMO对何时使用std :: forward以及何时使用std :: move进行了最佳解释。它提出了通用参考的概念,IMO是一个非常有用的推理移动语义的工具。