我读了一篇关于 C++11 中移动语义的漂亮文章。这篇文章写得很直观。文章中的示例类如下所示。
class ArrayWrapper
{
public:
// default constructor produces a moderately sized array
ArrayWrapper ()
: _p_vals( new int[ 64 ] )
, _metadata( 64, "ArrayWrapper" )
{}
ArrayWrapper (int n)
: _p_vals( new int[ n ] )
, _metadata( n, "ArrayWrapper" )
{}
// move constructor
ArrayWrapper (ArrayWrapper&& other)
: _p_vals( other._p_vals )
, _metadata( other._metadata )
{
other._p_vals = NULL;
}
// copy constructor
ArrayWrapper (const ArrayWrapper& other)
: _p_vals( new int[ other._metadata.getSize() ] )
, _metadata( other._metadata )
{
for ( int i = 0; i < _metadata.getSize(); ++i )
{
_p_vals[ i ] = other._p_vals[ i ];
}
}
~ArrayWrapper ()
{
delete [] _p_vals;
}
private:
int *_p_vals;
MetaData _metadata;
};
显然,在上述移动构造函数实现中,嵌入元素不会发生移动_metadata
。为了促进这一点,诀窍是使用这样的std::move()
方法。
ArrayWrapper (ArrayWrapper&& other)
: _p_vals( other._p_vals )
, _metadata( std::move( other._metadata ) )
{
other._p_vals = NULL;
}
到目前为止,一切都很好。
标准说:
§5(C++11 §5[expr]/6):
[ 注意:表达式是一个 xvalue,如果它是:
调用函数的结果,无论是隐式还是显式,其返回类型是对对象类型的右值引用,
转换为对对象类型的右值引用,
一个类成员访问表达式,指定一个非引用类型的非静态数据成员,其中对象表达式是一个 xvalue,或
一个
.*
指向成员的表达式,其中第一个操作数是一个 xvalue,第二个操作数是一个指向数据成员的指针。
我的问题:
现在,other
移动构造函数中的变量是一个 xvalue(对吗?)。那么根据上面最后一条规则,other._metadata
也应该是一个xvalue。因此编译器可以隐式使用_metadata
' 类的移动构造函数。所以,这里没必要std::move
。
我错过了什么?