C++中的左值和右值

简介:

在C/C++中,左值(lvalue)和右值(rvalue)是用于规定表达式(expression)的性质。C++中表达式要不然是左值,要不然是右值。

这两个概念在C语言中比较容易理解:左值能放在赋值语句的左边,右值不能。但是当来到C++时,二者的理解就比较复杂了(PS:有对象真是麻烦)

简单的归纳:

  • 当一个对象被用作右值的时候,用的是对象的值(内容);当对象被用作左值的时候,用的是对象的身份即在内存中的地址。
  • 左值是代表一个内存地址值,并且通过这个内存地址,就可以对内存进行读并且写(主要是能写)操作。
  • 在需要右值的地方可以用左值来代替,但是不能把右值当成左值使用。
  • 关键是搞清楚,什么是右值,或者说什么不能用作左值(字面常量、&a的结果等等)。

举例来说:

  1. 赋值运算符需要一个(非常量)左值作为其左侧运算对象,最后得到的结果也是一个左值。eg:
    num1 = num2 =num3;
    在这里,等于运算符从右到左计算,所以num2和num1是左值,num2=num3得到的结果也是左值,但是在这个语句里被当成右值使用了。
  2. 取地址符作用于一个左值运算对象,返回一个指向该运算对象的指针,这个指针是一个右值。eg:
    int a = 10; int *p = &a;
    在这里,p是一个左值,&a的结果是右值,因为&a的结果是一个地址(字面常量)。
  3. 内置解引用运算符、下标运算符、迭代器解引用运算符、string和vector的下标运算符的求值结果,都是左值。
  4. 内置类型和迭代器的递增递减运算符作用于左值运算对象所得的结果也是左值。

特例两个

  1. 当函数的返回值是引用类型是,可以用作左值,当函数的返回值是其他类型时,不能用作左值。eg:
    char &get_val(string &str, string::size_type ix )
    {
    return str[ix];
    }
    int main()
    {
    string s("a value");
    cout << s << endl;
    get_val(s,0) = 'A'; //get_val函数调用作为左值
    return 0;
    }
    在这里,get_val函数的返回值是引用,可以用作左值。
  2. 关于decltype,decltype返回的是一个类型不是一个表达式。当decltype作用于表达式的时候,结果根据表达式的不同也有所不同。如果表达式的结果是一个左值,这decltype作用于该表达式得到一个引用类型。eg:
    int a = 11;
    int *p = &a; 
    decltpye(*p) b = a; 


    如果表达式的结果是一个右值,则decltype作用于该表达式得到的是一个指针类型。eg:
    int a = 11;
    int *p = &a;
    decltype(&p) p2 = &p



相关文章
|
5月前
|
编译器 C++ 容器
【c++11】c++11新特性(上)(列表初始化、右值引用和移动语义、类的新默认成员函数、lambda表达式)
C++11为C++带来了革命性变化,引入了列表初始化、右值引用、移动语义、类的新默认成员函数和lambda表达式等特性。列表初始化统一了对象初始化方式,initializer_list简化了容器多元素初始化;右值引用和移动语义优化了资源管理,减少拷贝开销;类新增移动构造和移动赋值函数提升性能;lambda表达式提供匿名函数对象,增强代码简洁性和灵活性。这些特性共同推动了现代C++编程的发展,提升了开发效率与程序性能。
176 12
|
10月前
|
存储 安全 C++
【C++11】右值引用
C++11引入的右值引用(rvalue references)是现代C++的重要特性,允许更高效地处理临时对象,避免不必要的拷贝,提升性能。右值引用与移动语义(move semantics)和完美转发(perfect forwarding)紧密相关,通过移动构造函数和移动赋值运算符,实现了资源的直接转移,提高了大对象和动态资源管理的效率。同时,完美转发技术通过模板参数完美地转发函数参数,保持参数的原始类型,进一步优化了代码性能。
138 2
|
12月前
|
编译器 C++
C++ 11新特性之右值引用
C++ 11新特性之右值引用
112 1
|
编译器 C语言 C++
从C语言到C++_33(C++11_上)initializer_list+右值引用+完美转发+移动构造/赋值(中)
从C语言到C++_33(C++11_上)initializer_list+右值引用+完美转发+移动构造/赋值
113 1
从C语言到C++_33(C++11_上)initializer_list+右值引用+完美转发+移动构造/赋值(中)
|
存储 安全 C语言
从C语言到C++_33(C++11_上)initializer_list+右值引用+完美转发+移动构造/赋值(上)
从C语言到C++_33(C++11_上)initializer_list+右值引用+完美转发+移动构造/赋值
97 2
|
编译器 C语言 C++
从C语言到C++_33(C++11_上)initializer_list+右值引用+完美转发+移动构造/赋值(下)
从C语言到C++_33(C++11_上)initializer_list+右值引用+完美转发+移动构造/赋值
117 1
|
编译器 C++ 开发者
C++一分钟之-右值引用与完美转发
【6月更文挑战第25天】C++11引入的右值引用和完美转发增强了资源管理和模板灵活性。右值引用(`&&`)用于绑定临时对象,支持移动语义,减少拷贝。移动构造和赋值允许有效“窃取”资源。完美转发通过`std::forward`保持参数原样传递,适用于通用模板。常见问题包括误解右值引用只能绑定临时对象,误用`std::forward`,忽视`noexcept`和过度使用`std::move`。高效技巧涉及利用右值引用优化容器操作,使用完美转发构造函数和创建通用工厂函数。掌握这些特性能提升代码效率和泛型编程能力。
147 0
|
编译器 C++ 容器
【C++11(一)】右值引用以及列表初始化
【C++11(一)】右值引用以及列表初始化
|
存储 安全 程序员
C++11:右值引用
C++11:右值引用
95 0
|
存储 算法 程序员
【C++入门到精通】右值引用 | 完美转发 C++11 [ C++入门 ]
【C++入门到精通】右值引用 | 完美转发 C++11 [ C++入门 ]
119 0