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

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.xdnf.cn/news/15038.html

相关文章:

  • 小菜狗的云计算之旅,学习了解rsync+sersync实现数据实时同步(详细操作步骤)
  • PanoSAM:使用 Meta Segment Anything 模型对全景图像中的点云数据进行无监督分割
  • 深入理解 classnames:React 动态类名管理的最佳实践
  • 设计模式之迭代器模式
  • 基于改进多目标优化算法的众包物流配送调度研究
  • 【机器学习笔记Ⅰ】2 线性回归模型
  • 百度文心大模型 4.5 开源深度测评:技术架构、部署实战与生态协同全解析
  • 编程语言艺术:C语言中的属性attribute笔记总结
  • bro code笔记 inheritance
  • 从前端转nest.js开发的学习路线
  • 【Guava】1.0.设计虚拟机的方向
  • java学习——guava并发编程练习
  • Redis ①⑤-集群
  • Vue+Openlayers加载OSM、加载天地图
  • 周任务自动化升级:N8N与多维表格无缝联动全解析
  • 2025年03月 C/C++(四级)真题解析#中国电子学会#全国青少年软件编程等级考试
  • 【架构艺术】IC(个人贡献者)视角下产品研发规划的实战Tips
  • 一次内存“卡顿”全流程实战分析:从制造问题到优化解决
  • java中,stream的filter和list的removeIf筛选速度比较
  • JMM--数据原子操作
  • SpringAI与智能体入门
  • 解决kali Linux在VMware中的全局缩放问题
  • 量化可复用的UI评审标准(试验稿)
  • Python PyJWT详解:从入门到实战
  • 3dmax烘焙插件3dmax法线贴图烘焙教程glb和gltf元宇宙灯光效果图烘焙烘焙光影贴图支持VR渲染器
  • 针对工业触摸屏维修的系统指南和资源获取途径
  • NumPy-核心函数np.matmul()深入解析
  • CppCon 2018 学习:Surprises In Object Lifetime
  • 设计模式之访问者模式
  • 使用Langchain访问个人数据