c++左值 右值 左值引用 右值引用

左值、右值、左值引用、右值引用

  • 无法令引用重新绑定到另外一个对象上,因此引用必须初始化,引用并非对象,只是为一个已经存在的对象所起的另外一个名字
  • 因为引用本身不是一个对象,所以不能定义引用的引用
int a = 10;
int &b = a;
int &c = b;//此时c不是引用的引用,而是b c 都是a的引用
  • 判断某个表达式是左值还是右值的方法

    • 可以位于赋值号左侧的表达式就是左值;只能位于赋值号右侧的表达式就是右值

    • 有名称的,可以获取到存储地址的表达式为左值,反之为右值

  • 右值引用只能绑定到一个将要销毁的对象,必须进行初始化,只能使用右值来初始化

  • 一般一个左值表达式表示的是一个对象的身份,一个右值表达式表示的是对象的值

int i = 42;
int &r = i;//正确
//    int &&r = i;//错误,不能将一个右值引用绑定到一个左值上
//    int &r2 = i * 42; //错误,i * 42是一个右值
const int &r3 = i * 42; //正确,可以将一个const的引用绑定到一个右值上
int &&r2 = i * 42;//正确

右值引用的使用

  • 和声明左值引用一样,右值引用也必须立即进行初始化操作,而且只能使用右值进行初始化
int &&a = 10;
  • 右值引用可以修改右值
int &&a = 10;
cout<<a<<" "<<&a<<endl;
a = 11;
cout<<a<<" "<<&a<<endl;
  • 存在常量右值引用,但是这种表示没有任何实际用处。
    • 右值引用主要用于移动语义和完美转发,前者需要有修改右值的权限
    • 常量右值引用就是引用一个不可修改的右值,可以交给常量左值引用
const int&& a = 10;
  • 返回非引用类型的函数,连同算数、关系、位以及后置递增/递减运算符,都生成右值,不能将一个左值引用绑定到这类表达式上,但可以将一个const的左值引用或者一个右值引用绑定到这类表达式上
  • 由于右值引用只能绑定到临时对象,可以得知
    • 所引用的对象将要被销毁
    • 该对象没有其他用户
    • 意味着右值引用的代码可以自由地接管所引用的对象的资源

实例

#include <iostream>
using namespace std;
struct Foo
{
    Foo(){ cout << "Foo()" << endl; }
    Foo(const Foo&ref){ cout << "Foo(const Foo&)" << endl; } // copy ctor
    Foo(Foo&& ref){ cout << "Foo(Foo&&)" << endl; }          // move ctor
    Foo& operator=(const Foo& rhs){cout << "Foo& operator=(const Foo& rhs)" << endl; } // copy assignment
    Foo& operator=(Foo&& rhs){cout << "Foo& operator=(Foo&& rhs)" << endl; }           // move assignment
    ~Foo(){ cout << "~Foo()" << endl; }
};

int main(int argc, char* argv[])
{
    cout<<"=========="<<endl;
    Foo foo1 = Foo();//只调用了Foo()构造函数
    cout<<"=========="<<endl;
    foo1 = Foo();//先调用Foo()构造函数,再调用移动赋值运算符,之后调用析构函数
    cout<<"=========="<<endl;
    Foo  foo2 = foo1;//调用拷贝构造函数
    cout<<"=========="<<endl;
    foo2 = foo1;//调用拷贝赋值运算符
    getchar();
    return 1;
}

输出结果

==========
Foo()
==========
Foo()
Foo& operator=(Foo&& rhs)
~Foo()
==========
Foo(const Foo&)
==========
Foo& operator=(const Foo& rhs)
  • 实参类型为非常量右值引用则可以调用移动构造函数

总结

  • 非常量左值引用可以引用的值的只有非常量左值,常量左值引用可以引用非常量左值、常量左值以及右值
  • 右值引用不支持引用左值;非常量右值引用只可以引用非常量右值;常量右值引用可以引用非常亮右值引用和常量右值
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

孙建钊

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值