C++中的左值、右值、左值引用、右值引用

我们在定义变量或函数传参的时候编译器经常会提示“表达式必须是可以修改的左值”,“非常量引用的初始值必须是左值”等等报错,本文将对左值和右值,左值引用和右值引用做一个简单的总结。

1. 左值和右值

大多数情况下来说左值是=左边的值,右值是=右边的值,如

int a = 10

此时a为左值, 10为右值;

但是对于

int a = 10;

int b = a;

尽管a在=的右侧 ,但a仍为左值。

可以简要概括为在内存中有位置的变量为左值,而对于没有储存空间、没有位置的临时变量则为右值;我们可以给有储存位置左值赋值,但不能给没有储存位置的右值复制。

int a = 10;

20 = a; error:表达式必须是可以修改的左值

2. 左值引用和右值引用

很多情况下,为了节省内存,提高效率,函数参数值通常为引用,当我们不希望传递的引用被修改时可以加上const如

void func(const int &a){}

2.1 对于无const的引用

#include <iostream>

void func(int& a) {
    std::cout << a << std::endl;
}

int main() {
    {
        int a = 10;
        func(a);
        func(10); error: 非常量引用的初始值必须为左值
    }
}

可以看到现在我们只能给函数传递左值,即左值引用。

2.2 对于带const的情况

#include <iostream>

void func(const int& a) {
    std::cout << a << std::endl;
}

int main() {
    {
        int a = 10;
        func(a);
        func(10); //right
    }
}

当传递的参数为右值时,编译器会帮助我们生成一个临时变量,该情况下可以传递左值和右值。

2.3 右值引用

#include <iostream>

void func(int&& a) {
    std::cout << a << std::endl;
}

int main() {
    {
        int a = 10;
        func(a); error: 无法将右值引用绑定到左值
        func(10); 
    }
}

此时,我们只能传递右值(临时变量),即右值引用。

右值引用的一个好处是传递的是临时变量,它的作用和存在时间有限,在上面的例子中他只存在于func(10)的作用域内,因此你可以随意的操作它;但对于左值引用,它可能被多个函数使用,因此不能随意操作。

### C++左值引用引用的概念及区别 #### 定义与特性 在 C++ 中,表达式的分类对于理解引用至关重要。左值 (lvalue) 表达式通常表示具有持久状态的对象,而 (rvalue) 则指临时对象或字面量。 - **左值引用 (`T&`)**:只能绑定到左值,允许修改所引用的对象。 - **引用 (`T&&`)**:专门设计来绑定到,支持移动语义和完美转发[^1]。 #### 使用场景对比 | 类型 | 非常量左值引用 `T&` | 常量左值引用 `const T&` | 非常量引用 `T&&` | | --- | --- | --- | --- | | 绑定至左值 | 支持 | 支持 | 不支持 | | 绑定至 | 不支持 | 支持 | 支持 | 非常量左值引用主要用于传递参数以便函数内部能改变调用者提供的变量;常量左值引用则广泛应用于避免大对象复制开销的同时保持只读访问权限。特别地,在现代 C++ 编程实践中,非常量引用主要服务于两个目的: - 实现高效的资源转移——即所谓的“移动语义”,通过直接接管现有资源而非深拷贝; - 协助模板元编程中的“完美转发”。 ```cpp // 移动语义示例 class Widget { public: // 拷贝构造函数 Widget(const Widget &other); // 移动构造函数 Widget(Widget &&other) noexcept; }; std::vector<Widget> v; v.push_back(Widget()); // 调用了移动构造函数而不是拷贝构造函数 ``` #### 特殊情况说明 得注意的是,虽然常规情况下引用不允许绑定到左值上,但在特定条件下可以通过标准库提供的 `std::move()` 函数显式转换为可被引用捕获的状态。这并不意味着原对象真正变成了,而是告诉编译器该对象即将废弃,从而启用潜在的性能优化措施[^2]。 ```cpp int a = 42; int&& rref = std::move(a); // 显式将左值转成引用可以接受的形式 ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值