当前位置: 首页 > news >正文

C++ 中左值和右值

C++ 左值与右值全面讲解


一、什么是左值(Lvalue)与右值(Rvalue)

项目左值(Lvalue)右值(Rvalue)
是否有名字✅ 有,具名变量❌ 无名,通常是表达式/字面值
是否可取地址✅ 可以 &a❌ 不能 &(a + b)
生命周期通常在作用域内持续临时值,表达式计算后可能立即销毁
是否能赋值✅ 可以放在赋值号左边❌ 不可以出现在赋值号左边
举例int a;, a = 5;, int& r = a;10, a + b, getTemp()(非引用返回)

二、左值与右值基础示例

int a = 5;       // a 是左值
int b = a + 3;   // a + 3 是右值(表达式结果)

理解:

  • a 是具名变量,有内存地址,可被赋值;
  • a + 3 是右值,只存在于表达式计算期间。

三、引用绑定规则示例

int a = 10;
int& ref = a;          // ✅ 左值引用,绑定 a
ref = 15;              // ✅ 修改了 a 的值int& ref2 = 10;        // ❌ 错误!右值不能用普通引用绑定
const int& cref = 10;  // ✅ 合法!const 引用可绑定右值

结论:

  • 右值可以用 const T& 绑定;
  • 不能用普通 T& 来绑定右值;
  • const T& 会延长右值生命周期。

四、右值引用(C++11)

右值引用的语法:

int&& rref = 5;       // 右值引用
rref = 10;            // ✅ 修改右值绑定的变量

应用:用于移动语义,替代深拷贝,提高性能。


五、函数参数中的左值引用与右值引用重载

void print(string& s)      { cout << "左值引用: " << s << endl; }
void print(string&& s)     { cout << "右值引用: " << s << endl; }int main() {string a = "Hello";print(a);                  // 左值引用print(string("World"));    // 右值引用print(a + "!");            // 右值引用(临时表达式)
}

输出:

左值引用: Hello
右值引用: World
右值引用: Hello!

六、函数返回值与左值右值

int& getA() {static int a = 10;return a;
}int getB() {return 20;
}int main() {getA() = 30;    // ✅ 合法,getA() 是左值引用getB() = 40;    // ❌ 错误,getB() 是右值(返回值拷贝)
}

原则:

  • 返回 引用(左值) 可以继续赋值;
  • 返回 值(右值) 不能被赋值。

七、引用折叠(C++11)

用于模板中区分左值右值:

template<typename T>
void func(T&& arg);func(a);      // T = int&,T&& = int&(折叠为左值引用)
func(10);     // T = int,T&& = int&&(右值引用)

折叠规则口诀:

  • & &&
  • & &&&
  • && &&
  • && &&&&

八、const 修饰的指针/引用绑定

1. 右值绑定常量引用

const int& r = 100;   // OK,右值可以用 const 左值引用绑定

2. 左值引用右值

int& r = 100;         // ❌ 错误

九、右值引用与移动构造函数

class MyString {string data;
public:MyString(string&& s) : data(std::move(s)) {}
};

std::move(s) 将右值传递到 data,避免了复制构造。


十、配套总结表格

声明含义
int& a = b;左值引用,b 必须是左值
const int& a = 10;常量左值引用,可绑定右值
int&& a = 10;右值引用,绑定临时值或移动资源
int x = get(); x = 10;正常赋值
get() = 10;❌ 若 get() 返回右值,非法赋值
getRef() = 10;✅ 若返回引用,是左值,可赋值

十一、小测试题与解析

题 1:哪些是合法声明?

int a = 5;
int& r1 = a;             // ✅
int& r2 = 5;             // ❌,5 是右值
const int& r3 = 5;       // ✅,右值 + const
int&& r4 = 5;            // ✅,右值引用

题 2:以下调用是否合法?

int& get();    // 返回左值引用
int  set();    // 返回右值get() = 10;    // ✅ 合法
set() = 20;    // ❌ 错误,右值不能赋值

总结口诀回顾

“能取地址是左值,临时结果是右值;加 const 能绑定,双&&能转移。”

http://www.dtcms.com/a/268167.html

相关文章:

  • 论文评价指标之(n-gram、BLEU、MRR、ANLS)
  • python库 maya 库的各种案例的使用详解(人性化的日期时间处理)
  • 使用Python将PDF转换成word、PPT
  • SSL 终结(SSL Termination)深度解析:从原理到实践的全维度指南
  • 电商系统二次开发找谁做?ZKmall开源商城前后端分离技术更易升级迭代
  • leetcode 每日一题 1865. 找出和为指定值的下标对
  • python学习打卡:DAY 21 常见的降维算法
  • 红宝书学习笔记
  • 多级缓存如何应用
  • YOLO目标检测数据集类别:分类与应用
  • Oracle使用SQL一次性向表中插入多行数据
  • NLP之文本纠错开源大模型:兼看语音大模型总结
  • 李宏毅genai笔记:推理
  • Maven引入第三方JAR包实战指南
  • 支持向量机(SVM)在肝脏CT/MRI图像分类(肝癌检测)中的应用及实现
  • Python11中创建虚拟环境、安装 TensorFlow
  • AI编程:打造炫酷多语倒计时器
  • 【Elasticsearch】自定义评分检索
  • 评论区实现 前端Vue
  • 【openp2p】 学习4: 纳秒级别的时间同步算法及demo
  • 数学建模的一般步骤
  • FastAPI+React19开发ERP系统实战第04期
  • Hadoop YARN 命令行工具完全指南
  • ProCCD复古相机:捕捉复古瞬间
  • uniapp的光标跟随和打字机效果
  • LangChain有中文网可以访问,还有教程可以学
  • 手机FunASR识别SIM卡通话占用内存和运行性能分析
  • Jailer:一款免费的数据库子集化和数据浏览工具
  • ragflow本地部署教程linux Ubuntu系统
  • Android studio在点击运行按钮时执行过程中输出的compileDebugKotlin 这个任务是由gradle执行的吗