Page's Personal Website

右值引用的作用

2019-08-29
c++

[TOC]

右值引用

自从引入了这个右值引用之后,我们把之前通常叫的引用称为左值引用。

  • 不同

左值引用:我们之前所说的别名

右值引用:大部分时间和左指引用一样,但是有一点区别,他能够绑定到临时变量(右值)

A a;
A&& a_rref = a;
A&& temp_rref = A();
  • 作用

(1)避免拷贝,提高性能,实现move()

(2)避免重载参数的复杂性,实现forward()

Move Semantics

第一个作用,就不多说了

Perfect Forwarding

第二个作用,参考[2]

// 1.struct A没有const引用的构造函数
class A {
public:
    A(int a) : num(a){}
    A(A& a) : num(a.num) {}
    //A(const A& a) : num(a.num) {}
    A& operator=(A const & rhs) { return *this; }; // classical implementation
    A& operator=(A&& rhs);
private:
    int num;
};
// 2. factory 版本
// factory1
template <class T>
std::shared_ptr<T>
factory()   // no argument version
{
    return std::shared_ptr<T>(new T);
}
// factory2
//template <class T, class A1>
//std::shared_ptr<T>
//factory(const A1& a1)   // one argument version
//{
//    return std::shared_ptr<T>(new T(a1));
//}
// factory3
//template <class T, class A1>
//std::shared_ptr<T>
//factory(A1& a1)
//{
//    return std::shared_ptr<T>(new T(a1));
//}
// factory4
template <class T, class A1>
std::shared_ptr<T>
factory(A1&& a1)
{
    return std::shared_ptr<T>(new T(std::forward<A1>(a1)));
}

// 3.编译报错
            A* q = new A(5);
    std::shared_ptr<A> p = factory<A>(5);
    p = factory<A>(*q);

在calss A没有const引用的拷贝构造函数的情况下,结果是:

(1)如果factory的const引用参数这个版本(factory2)没有的话,那么std::shared_ptr<A> p = factory<A>(5);就会报如下错

error C2664: 'std::shared_ptr<_Ty> factory<A,int>(A1 &)' : cannot convert parameter 1 from 'int' to 'int &'

(2)如果factory的非const引用参数这个版本(factory3)没有的化,那么p = factory<A>(q);*这句就会报错

error C2558: class 'A' : no copy constructor available or copy constructor is declared 'explicit'

所以,这里就引出了一个问题,如果factory的参数有很多的化,那么重载函数的数量就指数增长了。右值的作用就发挥在这里了,factory4就只需要一个就搞定了

参考

[1]rvalue references

[2]A Brief Introduction to Rvalue References


Similar Posts

Comments