C++右值引用

C++右值引用

什么是右值

判断其为左值还是右值并不能单纯通过其在等号的左边还是右边
通俗的定义为左值是非临时对象,在表达式结束后依然存在,而右值为临时对象,仅对当前表达式有效。

定义int a=1; 判断1,a,a++,++a是左值还是右值?

1是立即数,是临时对象,为右值。
a在表达式结束后依然存在,不是临时对象,为左值。
a++是取出a的拷贝,然后对a加一,返回拷贝,是临时对象,为右值。
++a是对a加一,然后返回a,不是临时对象,为左值。

左值的声明符号为”&”,右值的声明符号为”&&”。
可以通过下面的代码,来对于上述结果进行验证。

template <class T> void print(T & t)
{
    cout << "LValue: " << t << endl;
}

template <class T> void print(T && t)
{
    cout << "RValue: " << t << endl;
}

int main()
{
    print(1);
    int a = 1;
    print(a);
    print(a++);
    print(++a);

    return 0;
}

输出结果为

RValue: 1
LValue: 1
RValue: 1
LValue: 3

右值引用的意义

右值引用的意义是为了解决对于不必要临时对象的生成与拷贝而造成的效率降低的问题。
下面给出一个例子

class MyStr
{
private: 
    char* str;

public:
    MyStr()
    {
        cout << "MyStr()" << endl;
        str = NULL;
    }

    MyStr(const char* s)
    {
        cout << "MyStr(const char*)" << endl;
        str = new char[strlen(s) + 1];
        strcpy(str, s);
    }

    MyStr(const MyStr &s)
    {
        cout << "MyStr(const MyStr&)" << endl;
        str = new char[strlen(s.str) + 1];
        strcpy(str, s.str);
    }

    MyStr& operator =(const MyStr &s)
    {
        cout << "MyStr& operator=(const MyStr&)" << endl;
        str = new char[strlen(s.str) + 1];
        strcpy(str, s.str);
        return *this;
    }

    ~MyStr()
    {
        cout << "~MyStr()" << endl;
        if (str)
        {
            delete[] str;
            str = NULL;
        }
    }

};

int main()
{
    MyStr a;
    a = MyStr("2");
}

输出结果为

MyStr()
MyStr(const char*)
MyStr& operator=(const MyStr&)
~MyStr()
~MyStr()

可以看出,采用传统方法在执行a = MyStr(“2”);步骤时,先申请存储空间,写入“2”,再在a中申请存储空间,调用strcpy,将“2”拷贝到a中申请的存储空间,最后将临时存储空间释放。
这样对于临时存储空间的申请,再释放的过程,可以说是对于效率的一种损耗。试想,能否直接将写入“2”以后的临时存储空间交给a,而不再做申请存储空间,赋值再删除的工作。
由此引入了右值引用。

加入右值引用的代码

    MyStr(MyStr &&s)
    {
        cout << "MyStr(const MyStr&&)" << endl;
        str = s.str;
        s.str = NULL;
    }

    MyStr& operator =(MyStr &&s)
    {
        cout << "MyStr& operator=(const MyStr&&)" << endl;
        delete[] str;
        str = s.str;
        s.str = NULL;
        return *this;
    }

可以看出,其中的赋值操作由str = new char[strlen(s.str) + 1];strcpy(str, s.str);变为了str = s.str;使得拷贝操作由深拷贝变为浅拷贝,从而提高效率。
采用了右值引用后,输出结果为:

MyStr()
MyStr(const char*)
MyStr& operator=(const MyStr&&)
~MyStr()
~MyStr()

可以看出a = MyStr(“2”);步骤中,采用了右值引用的特性。

在使用右值引用时,还需要注意两点:
1. 右值不可以是常量,因为我们需要修改右值。
2. 右值的资源链接和标记必须修改。否则,右值的析构函数就会释放资源。转移到新对象的资源也就无效了。

move语义
std:move用于将左值引用转化为右值引用。可以利用右值引用的特性,优化效率。
如swap函数,交换两个参数的值。通常采用的方式是备份,再交叉赋值,是一种深拷贝的方式。

template <class T> swap(T& a, T& b)
{
    T t(a);
    a = b;
    b = t;
}

采用move语义,将左值引用转化为右值引用。

template <class T> swap(T& a, T& b)
{
    T t(std::move(a));
    a = std::move(b);
    b = std::move(t);
}

通过这种方式就避免了3次不必要的拷贝操作。


所有代码均通过Visual Studio 2015调试

C++ 引用C++11 引入的一个重要特性,它主要用于实现移动语义和完美转发,提高程序的性能和效率。以下是关于 C++ 引用的详细介绍: ### 左的概念 在 C++ 中,表达式可以分为左(lvalue)和(rvalue)。左是指可以取地址、有名字的表达式,通常表示一个持久的对象;是指不能取地址、没有名字的临时对象或字面量。例如: ```cpp int a = 10; // a 是左,10 是 ``` ### 左引用引用 - **左引用**:使用 `&` 符号声明,只能绑定到左。例如: ```cpp int main() { int a = 10; int &b = a; // 定义一个左引用变量 b = 20; // 通过左引用修改引用内存的 return 0; } ``` - **引用**:使用 `&&` 符号声明,只能绑定到。例如: ```cpp int main() { int&& cc = 0; // 引用 return 0; } ``` 需要注意的是,常量左引用 `const T&` 可以绑定到,这为函数参数传递提供了一定的灵活性 [^3]。 ### 引用的使用场景 #### 移动语义 移动语义允许将资源(如动态分配的内存)从一个对象转移到另一个对象,而无需进行昂贵的深拷贝操作。这通过移动构造函数和移动赋运算符实现。 **移动构造函数示例**: ```cpp class Buffer { public: Buffer(size_t size) : size_(size), data_(new int[size]) {} // 移动构造函数 Buffer(Buffer&& other) noexcept : size_(other.size_), data_(other.data_) { other.size_ = 0; other.data_ = nullptr; // 确保 other 处于有效状态 } ~Buffer() { delete[] data_; } private: size_t size_; int* data_; }; ``` **移动赋运算符示例**: ```cpp Buffer& operator=(Buffer&& other) noexcept { if (this != &other) { delete[] data_; // 释放现有资源 data_ = other.data_; size_ = other.size_; other.data_ = nullptr; other.size_ = 0; } return *this; } ``` #### 完美转发 完美转发允许函数模板将参数以原始的类别(左)传递给其他函数,避免不必要的拷贝或移动操作。这通过引用折叠规则和 `std::forward` 函数实现。 ### 引用的特殊情况 - 引用:如果需要将左当作处理,可以使用 `std::move` 函数。例如: ```cpp int f = 0; int&& ff = std::move(f); // 引用必须加 move ``` ### 总结 引用C++ 带来了移动语义和完美转发的能力,使得程序在处理临时对象和资源管理时更加高效。通过合理使用引用,可以避免不必要的拷贝操作,提高程序的性能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值