C++ lvalue(左值)和rvalue(右值)

lvalue(左值)和rvalue(右值)

昨天写代码遇见一个这样的错误:{ "cannot bind non-const lvalue reference of type ‘int&’ to an rvalue of type ‘int’",代码类似下边

class MyClass {
    int data;
public:
    MyClass(int& e):data(e){};
};

int main(){
    MyClass c(10);
    return 0;
}

编译器告诉我们:“不能将非const的lvalue引用 和 rvalue绑定”,里边有两个关键词:lvaluervalue

什么是lvalue(左值)

简单定义:内存中有确定存储地址的对象的表达式的值,可以是一个变量名:int a;,可以是一个赋值表达式a=b,或者字符串常量"cnblogs",总之,最简单鉴别的方法就是编译以下这个句子&(需要鉴别的表达式),能通过编译就是左值。

严格定义见:C++文档

左值有什么性质?

  1. 可以被取地址(&),上边我们就是用这个方法来鉴别左值。
  2. 可以被赋值,或者复制赋值;
    //赋值
    int a = 10;
    //复制赋值
    int a(10);
  3. 可以初始化左值引用,也就是我们常用的引用;
    int a=10;
    int& reference = a;

    这是一些常用的性质,更多性质见:C++文档

什么是rvalue(右值)

:非左值。

常见的有字面值(除了字符串常量),后置自增自减表达式a++;a--;

右值有什么性质?

  1. 不能被取地址

    int a = 1;
    int* p = &(a++); //报错,a++是右值,不能被取地址,而++a是左值
  2. 不能被赋值或者复制赋值
    42 = 10;//很明显会报错
  3. 可以初始化const左值引用 ,这就是我开头程序报错的原因!我的程序没有const
    class MyClass {
        int data;
    public:
        MyClass(int const& e):data(e){};  //这里的形参引用用const修饰就可以编译通过了!!!
    };
    
    int main(){
        MyClass c(10);
        return 0;
    }
  4. 可以初始化右值引用;因为右值引用已经超出我的知识范畴了,以后学到再来补充。

为什么右值可以初始化const左值引用

note:下边的引用都是说左值引用

我先来说一下为什么右值不可以初始化普通引用,引用实际上是对指针的封装,主要作用是修改这个指针指向的值,那根据右值的性质:不能被取地址,所以引用和右值绑定显然是非法的。

那为什么加上const就可以了呢?

这是因为将const引用绑定到右值时,编译器采取了一种妥协机制:编译器会为临时数据创建一个新的、无名的临时变量,并将临时数据放入该临时变量中,然后再将引用绑定到该临时变量。注意,临时变量也是变量,所有的变量都会被分配内存。所以说const引用实际上是和右值的一个copy绑定了。

参考:https://zh.cppreference.com/w/cpp/language/value_category#cite_note-3

http://c.biancheng.net/view/vip_2254.html

原文地址:https://www.cnblogs.com/rookiezjz/p/12350050.html

时间: 2024-10-05 03:09:38

C++ lvalue(左值)和rvalue(右值)的相关文章

C++ 11 左值,右值,左值引用,右值引用,std::move, std::foward

这篇文章要介绍的内容和标题一致,关于C++ 11中的这几个特性网上介绍的文章很多,看了一些之后想把几个比较关键的点总结记录一下,文章比较长.给出了很多代码示例,都是编译运行测试过的,希望能用这些帮助理解C++ 11中这些比较重要的特性. 关于左值和右值的定义 左值和右值在C中就存在,不过存在感不高,在C++尤其是C++11中这两个概念比较重要,左值就是有名字的变量(对象),可以被赋值,可以在多条语句中使用,而右值呢,就是临时变量(对象),没有名字,只能在一条语句中出现,不能被赋值. 在 C++1

翻译「C++ Rvalue References Explained」C++右值引用详解 Part5:右值引用就是右值吗?

本文为第五部分,目录请参阅概述部分:http://www.cnblogs.com/harrywong/p/cpp-rvalue-references-explained-introduction.html. 右值引用就是右值吗? 同之前一样,给出一个X类,让我们可以重载它的拷贝构造函数和拷贝赋值操作符来实现move语义.现在做如下考虑: void foo(X&& x) { X anotherX = x; // ... } 一个有趣的问题就是,在foo函数内,哪一个x的拷贝构造函数重载将会被

左值引用,右值引用以及移动语义

1.左值引用 左值引用的基本语法 Type &引用名 = 左值表达式; #include <iostream> using namespace std; int main() { int a = 10; //ra是a的引用(别名),相当于把ra与a绑定. int &ra(a); cout << ra << " " << (void*)&ra << " " << (void

C++ 11中的左值引用和右值引用

1.首先区分左值和右值    左值是表达式结束后依然存在的持久对象    右值是表达式结束时就不再存在的临时对象    便捷方法:对表达式取地址,如果能,则为左值,否则为右值举例:    int a = 10    int b = 20    int *pFlag = &a    vector<int> vctTemp    vctTemp.push_back(1)    string str1 = "hello"    string str2 = "wo

左值引用和右值引用

1.左值和右值的概念 左值是可以放在赋值号左边可以被赋值的值:左值必须要在内存中有实体:         右值当在赋值号右边取出值赋给其他变量的值:右值可以在内存也可以在CPU寄存器.         一个对象被用作右值时,使用的是它的内容(值),被当作左值时,使用的是它的地址. 2.引用 引用是C++语法做的优化,引用的本质还是靠指针来实现的.引用相当于变量的别名. 引用可以改变指针的指向,还可以改变指针所指向的值. 引用的基本规则: 声明引用的时候必须初始化,且一旦绑定,不可把引用绑定到其他

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

1.左值和右值的概念 左值是可以放在赋值号左边可以被赋值的值:左值必须要在内存中有实体:         右值当在赋值号右边取出值赋给其他变量的值:右值可以在内存也可以在CPU寄存器.         一个对象被用作右值时,使用的是它的内容(值),被当作左值时,使用的是它的地址. 2.引用 引用是C++语法做的优化,引用的本质还是靠指针来实现的.引用相当于变量的别名. 引用可以改变指针的指向,还可以改变指针所指向的值. 引用的基本规则: 声明引用的时候必须初始化,且一旦绑定,不可把引用绑定到其他

深入学习c++--左值引用和右值引用

#include <iostream> #include <string> #include <vector> using namespace std; int main() { int t = 10; //t: 左值 int t2 = t + 1; //t: 右值 int a = 1; const int &b = a + 1; // 左值引用 // int &b = a + 1; // 错误 cout << b << &quo

C++左值引用和右值引用

1. 左值:一般指的是可以取地址.有名字的.反之就是右值. 2. 标准上来说,赋值运算符返回的是lvalue refering to left operand,即左值引用. 3. cosnt int & i = 1;合法,但是int& i = 1;不合法,因为右值不可以绑定到一个引用上,但是可以绑定到const 引用上. 事实上也很好理解,我们肯定不能修改字面值1,因为它就是固定的值1,因而也必须绑定到const 引用上.

error C2662 无法将左值绑定到右值 —— 变量永远是左值,即使它的类型为右值引用

1 #include <utility> 2 3 struct A 4 { 5 void f() && { } 6 }; 7 8 void g(A &&a) 9 { 10 a.f(); //error, a是左值 11 std::move(a).f(); //fine 12 } 13 14 int main() { return 0; } 其实有了右值表示临时对象这一观察结果,变量是左值这一特性并不令人惊讶.毕竟,变量是持久的,直到离开作用域时才被销毁.(<

VS2012 error C2664: “std::make_pair”:无法将左值绑定到右值引用

例如:_mapTransportInfos.insert(std::make_pair<uint32, CTransportInfoPtr>(iter_t->m_nID, pinfo));这句代码在vs2012的c++11就无法编译,报2664错误.可以用如下强制转换来实现: 第一种方法(我这没成功)_mapTransportInfos.insert(std::make_pair<uint32, CTransportInfoPtr>(static_cast<uint32