Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

std::move()的原理 #50

Open
nine-point-eight-p opened this issue May 6, 2022 · 0 comments
Open

std::move()的原理 #50

nine-point-eight-p opened this issue May 6, 2022 · 0 comments

Comments

@nine-point-eight-p
Copy link

std::move()的原理

你真的了解std::move()吗?(我知道我不

我们都知道std::move()可以把左值转换为右值,然后就可以方便地使用move类的东西了。但是有时候std::move()的行为好像比较奇怪。以第三次作业第二题(引用?复制?)为例:

class Test {
    int *buf;
public:
    Test() {
        buf = new int(0);
        cout << "Test(): this->buf @ " << hex << buf << endl;
    }
    Test(int val) {
        buf = new int(val);
        cout << "Test(int): this->buf @ " << hex << buf << endl;
    }
    ~Test() {
        cout << "~Test(): this->buf @ " << hex << buf << endl;
        if (buf) delete buf;
    }
    Test(const Test& t) : buf(new int(*t.buf)) {
        cout << "Test(const Test&) called. this->buf @ "
            << hex << buf << endl;
    }
    Test(Test&& t) : buf(t.buf) {
        cout << "Test(Test&&) called. this->buf @ "
            << hex << buf << endl;
        t.buf = nullptr;
    }
    Test& operator= (const Test& right) {
        cout << "operator=(const Test&) called. this->buf @ "
            << hex << buf << endl;
        if (this != &right){
            if(buf) delete buf;
            buf = new int(*right.buf);
        }
        return *this;
    }
    Test& operator= (Test&& right) {
        cout << "operator=(Test&&) called. this->buf @ "
            << hex << buf << endl;
        if (this != &right){
            if(buf) delete buf;
            this->buf = right.buf;
            right.buf = nullptr;
        }
        return *this;
    }
    void print(const char *name) const {
        cout << name << ".buf @ " << hex << buf << endl;
    }
};

Test func(const Test& t) {
    Test b = std::move(t); // HERE
    return b;
}

int main() {
    Test a = std::move(func(3));
    return 0;
}

运行一下就可以知道,HERE处实际上调用的是拷贝构造函数。为什么不调用移动构造函数?

其实上一届的同学也讨论过这个问题,采用了实验的方法,参见关于std::move()到底干了什么? · Issue #32 · thu-coai/THUOOP · GitHub。这次我们做些理论层面的工作,来看一看std::move()究竟是怎么实现的。

模板的特化

如果已经学习了这部分知识或不感兴趣可以跳过这一节。

首先简要介绍一下模板的特化(specialization)。顾名思义,特化就是模板针对某些特殊情形的特别处理,也就是对某些特别的类型参数的处理。为什么要特化?因为对于特定的情况,如果你能给出更合适的实现,那么当然就该用你提供的。

模板分为类模板和函数模板,而特化分为全特化和偏特化。全特化就是完全限定模板要用的参数,偏特化就是部分限定参数。例如:

template<typename T1, typename T2>
class Test
{
public:
	Test(T1 i,T2 j):a(i),b(j){ cout<<"模板类"<<endl; }
private:
	T1 a;
	T2 b;
};
 
template<>
class Test<int, char>
{
public:
	Test(int i, char j):a(i),b(j){ cout<<"全特化"<<endl; }
private:
	int a;
	char b;
};
 
template <typename T2>
class Test<int, T2>
{
public:
	Test(int i, T2 j):a(i),b(j){ cout<<"偏特化"<<endl; }
private:
	int a;
	T2 b;
};

第一个是基本的函数模板,有两个类型参数T1T2。第二个和第三个都是其特化。其中第二个是全特化,类型名Test后面的<int, char>限定T1intT2char,所以template后面的<>里是空的(没有variable了)。第三个是偏特化,类型名Test后只把T1限定为int,而T2仍保留,所以template后面的<>里还有T2

特化如何工作呢?看如下例子:

Test<double, double> t1(0.1, 0.2); // 使用基本模板,无特化
Test<int, char> t2(1, 'A'); // 使用全特化版本
Test<int, bool> t3(2, true); // 使用偏特化版本

这是Test类的实例化(instantiation,不要与特化混淆)。其实就是个匹配的过程。编译器先看能不能用特化的版本,如果不能的话就还是用基础的版本。理应如此。

此外,类模板可以全特化、偏特化,而函数模板只能全特化(其他的可以通过重载实现)。这已经不是本文需要的内容了,我们只需要对特化的工作方式有个感性认识就可以继续了。

std::move()的实现

如果使用vscode,按住ctrl单击std::move()即可跳转至其实现。上代码:

// move.h
/**
  *  @brief  Convert a value to an rvalue.
  *  @param  __t  A thing of arbitrary type.
  *  @return The parameter cast to an rvalue-reference to allow moving it.
  */ 
template<typename _Tp>
  constexpr typename std::remove_reference<_Tp>::type&&
  move(_Tp&& __t) noexcept
  { return static_cast<typename std::remove_reference<_Tp>::type&&>(__t); }

……有点复杂?其实没有看上去那么复杂。可以拆出以下几个部分:

  • template<typename _Tp>:这是模板函数的标志。_Tp是类型参数。
  • constexpr typename std::remove_reference<_Tp>::type&&:这是返回值类型。constexpr表示编译期常量,可以暂时忽略。typename指示编译器,接下来的东西是个类型名。真正的返回值类型其实就是std::remove_reference<_Tp>::type,也就是std::remove_reference<_Tp>这个类里面定义的type
  • move(_Tp&& __t):这是函数名及其参数。noexcept表示不会抛出异常。
  • 函数体:只有一行,直接返回。可以看出,是要对参数__t进行一个static_cast,目标类型是刚才提到过的std::remove_reference<_Tp>::type&&

由此可见,std::move()实现的关键在于std::remove_reference<_Tp>

*若不感兴趣可以跳过此部分。*它的源码可以是这样的:

// type_traits
// remove_reference
template<typename _Tp>
  struct remove_reference
  { typedef _Tp   type; };

template<typename _Tp>
  struct remove_reference<_Tp&>
  { typedef _Tp   type; };

template<typename _Tp>
  struct remove_reference<_Tp&&>
  { typedef _Tp   type; };

第一个是基本模板,而第二个和第三个则是它的特化。特化在这里便发挥了像“选择”一样的作用,决定了std::remove_reference的具体实现。如果是普通的_Tp,那么就用第一个;如果是_Tp&,即_Tp的左值引用,就用第二个;同理,_Tp&&就用第三个。然而,无论哪个版本,都用typedef_Tp定义为type。因此,如果传进普通的_Tptype毫无疑问是_Tp;如果传进来的是带引用(&)的_Tp,无论是左值引用还是右值引用,模板推导规则会把引用从_Tp上“剥离”,从而type还是_Tp原本的样子。由此便达到了所谓remove_reference的效果。因为这都是基于模板推导实现的,所以以上均在编译期即可完成。

*跳到这里 ~* 回过头看move便一目了然。std::remove_reference正如其名,只是去掉了引用标签,那么所谓的std::remove_reference<_Tp>::type&&,其实就是两步:先把_Tp上的引用标签都去掉,取出这个“纯粹”的类型,再强行打上一个右值引用的标签,就得到了目标类型。然后由static_cast转换,一步到位。

于是我们也就更加清楚地了解了std::move()的功能:std::move()只是修改成右值引用。具体地说,T&改成T&&T&&还是T&&const T&改成const T&&const T&&还是const T&&。其实源码上面的注释也说了:The parameter cast to an rvalue-reference to allow moving it. 应当注意,std::move()和"move"(包括移动构造、移动赋值)没有直接的关系。如果类并没有相应的移动函数,std::move()也做不了什么。毕竟它只是"to allow moving it"。

至于本文开头的问题,也就很好解释了。std::move()接受一个const T&,出来的就是const T&&。如果构造函数的参数只接受const T&(拷贝)和T&&(移动)型,那么const T&&就只能按万能的const T&来调用拷贝构造,毕竟不能随随便便把const的东西当成非const处理。

杂谈

一些可以进一步研究的东西:

  • 左值/右值:lvalue?rvalue?prvalue?xvalue?尝试区分。
  • 模板元编程(Template Metaprogramming, TMP):如果你对C++模板的这些奇形怪状的花样感兴趣,可以尝试。(一言不合就千百行报错,你值得拥有

参考文献

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

1 participant