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

在 C++ 中,左值(lvalue)和右值(rvalue)是指对象的不同类别,区分它们对于理解 C++ 中的表达式求值和资源管理非常重要,尤其在现代 C++ 中涉及到移动语义(Move Semantics)和完美转发(Perfect Forwarding)时。

一、左值(Lvalue)

1. 定义

左值(lvalue)表示的是一个有名称的、持久的内存位置,可以在表达式的左侧,也可以在右侧使用。简单来说,左值是可以通过引用来访问的对象,它在程序的生命周期中有一个持久的存储位置。

2. 特征

  • 可修改:大多数左值是可修改的(当然,常量左值不可修改)。
  • 有持久地址:左值有内存地址,可以通过 & 取地址。

3. 例子

int x = 10;   // x 是一个左值,代表存储它的内存位置
x = 20;       // x 作为左值出现在赋值表达式的左侧

4. 常见的左值类型

  • 普通变量:如 int x = 10; 中的 x
  • 数组元素:如 arr[3]
  • 对象成员:如 obj.member
  • 解引用指针:如 ptr

二、右值(Rvalue)

1. 定义

  • 右值(rvalue)表示的是临时对象或不具有持久内存位置的对象,通常是表达式的结果。右值可以出现在赋值表达式的右侧,但不能出现在左侧(除非作为右值引用)。

2. 特征

  • 无持久地址:右值通常是一个临时对象,它在某些情况下会被销毁。
  • 不能修改:右值本身不表示一个持久的存储位置,所以不能被赋值或取地址。

3. 例子

int x = 10;  // x 是左值
int y = 20;  // y 是左值
y = x + 5;   // (x + 5) 是右值,表示一个临时结果

4. 常见的右值类型

  • 字面量:如 53.14'a' 等。
  • 临时对象:如表达式的返回值,例如 x + y 返回一个临时结果。
  • 函数返回值:如返回一个非引用的临时值 int foo() { return 42; }
  • 类型转换:如 (int)3.14 或 std::move(x)

三、右值引用(Rvalue Reference)

C++11 引入了右值引用(T&&),使得我们能够有效地使用右值(临时对象)进行资源转移(例如移动语义)。右值引用允许对象的资源不需要复制,而是可以直接“移动”到新的对象中,这样能提高程序的效率,避免不必要的资源复制。

1. 右值引用的使用

  • 右值引用的声明通常为 T&&(例如,int&&)。
  • std::move 将一个左值转换为右值引用,从而启用移动语义。
  • 移动构造函数和移动赋值运算符通常采用右值引用作为参数,允许从临时对象中“偷取”资源。

2. 示例

#include <iostream>#include <vector>void printVector(std::vector<int>&& v) {for (auto i : v) {std::cout << i << " ";}std::cout << std::endl;
}int main() {printVector({1, 2, 3, 4, 5});  // 使用右值传递临时对象return 0;
}

在这个例子中,{1, 2, 3, 4, 5} 是一个右值,可以作为右值引用参数传递给 printVector 函数。

四、区别与联系

  • 左值(Lvalue):表示一个持久的对象,它有地址并且可以被修改。例如,变量、数组元素、解引用指针等。
  • 右值(Rvalue):表示一个临时对象或没有持久地址的值,通常出现在赋值语句的右边。它们通常不可修改。
  • 右值引用(Rvalue Reference):C++11 引入的一种新类型,允许我们将右值传递给函数,从而避免资源的复制(通过移动语义)。右值引用通过 T&& 来表示。

五、C++11 中的扩展:完美转发和移动语义

1. 完美转发(Perfect Forwarding)

通过右值引用,我们可以将函数的参数完美地转发到另一个函数,无论是左值还是右值。使用 std::forward 可以实现完美转发。

template<typename T>
void wrapper(T&& arg) {func(std::forward<T>(arg));  // 完美转发 arg 到 func 函数
}

2. 移动语义(Move Semantics)

右值引用可以用于“移动”资源,而不是复制它们。移动构造函数和移动赋值运算符允许通过“转移”资源来避免不必要的内存复制。

std::vector<int> getVector() {std::vector<int> v = {1, 2, 3, 4};return v;  // 返回一个右值
}int main() {std::vector<int> v = getVector();  // 通过移动语义,避免了不必要的复制
}

六、总结

  • 左值(Lvalue):有持久存储位置,通常表示变量或对象。
  • 右值(Rvalue):没有持久存储位置,通常表示临时对象或值。
  • 右值引用(Rvalue Reference):用于支持移动语义,允许我们移动而不是复制资源。

通过区分左值和右值,C++ 提供了更高效的内存管理方式,尤其在现代 C++ 中,移动语义和完美转发能够显著提高性能。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/35436.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

优选算法系列(2.滑动窗口 _ 上)

目录 解法⼀&#xff08;暴力求解&#xff09;&#xff08;不会超时&#xff0c;可以通过&#xff09;&#xff1a;一.长度最小的子数组&#xff08;medium&#xff09; 题目链接209. 长度最小的子数组 - 力扣&#xff08;LeetCode&#xff09; 解法&#xff1a; 代码&#…

ELK(Elasticsearch、Logstash、Kbana)安装及Spring应用

Elasticsearch安装及Spring应用 一、引言二、基本概念1.索引&#xff08;Index&#xff09;2.类型&#xff08;Type&#xff09;3.文档&#xff08;Document&#xff09;4.分片&#xff08;Shard&#xff09;5.副本&#xff08;Replica&#xff09; 二、ELK搭建1.创建挂载的文件…

Redis,从数据结构到集群的知识总结

Redis基础部分 2. 数据结构 redis底层使用C语言实现&#xff0c;这里主要分析底层数据结构 2.1 动态字符串(SDS) 由于C底层的字符串数组一旦遇到’\0’就会认为这个字符串数组已经结束&#xff0c;意味着无法存储二进制数据&#xff08;如图片、音频等&#xff09;&#xff…

【redis】Jedis 操作 Redis 基础指令(下)

列表操作 lpush/rpush 和 lpop/rpop 将一个或者多个元素从左/右侧放入&#xff08;头/尾插&#xff09;到 list 中 依次头插 从 list 左/右侧取出元素&#xff08;即头/尾删&#xff09; public static void test1(Jedis jedis) { jedis.flushAll(); long n jedis.lpush(…

基于消失点标定前视相机外参

1. 消失点 艺术家&工程师在纸上表现立体图时,常用一种透视法,这种方法源于人们的视觉经验:近大远小,且平行的直线都消失于无穷远处同一个点。就像我们观察两条平行的铁轨时会觉得他们相交于远处的一点,我们把这个点称为消失点。 图1 铁轨组成的消失点 2. 在标定中的应…

TypeScript接口 interface 高级用法完全解析

TypeScript接口 interface 高级用法完全解析 mindmaproot(TypeScript接口高级应用)基础强化可选属性只读属性函数类型高级类型索引签名继承与合并泛型约束设计模式策略模式工厂模式适配器模式工程实践声明合并类型守卫装饰器集成一、接口核心机制深度解析 1.1 类型兼容性原理 …

Vue3 Pinia $subscribe localStorage的用法 Store的组合式写法

Vue3 Pinia $subscribe 可以用来监视Stroe数据的变化 localStorage的用法 localStorage中只能存字符串&#xff0c;所有对象要选转成json字符串 定义store时&#xff0c;从localStorage中读取数据talkList可能是字符串也可能是空数组 Store的组合式写法 直接使用reactiv…

新版AndroidStudio / IDEA上传项目到Gitee

目录 1.Gitee创建仓库 2.填写仓库的信息 3.创建成功后复制仓库的地址 4.检查AndroidStudio是否配置Git 5.点击测试 6.之后Create Git Repository 7.添加到本地仓库 8.提交项目 9.添加上传仓库的地址 10.上传成功 11.去Gitee上刷新检查 1.Gitee创建仓库 2.填写仓库的…

用 Vue 3.5 TypeScript 重新开发3年前甘特图的核心组件

回顾 3年前曾经用 Vue 2.0 开发了一个甘特图组件&#xff0c;如今3年过去了&#xff0c;计划使用Vue 3.5 TypeScript 把组件重新开发&#xff0c;有机会的话再开发一个React版本。 关于之前的组件以前文章 Vue 2.0 甘特图组件 下面录屏是是 用 Vue 3.5 TypeScript 开发的目前…

C语言【数据结构】:时间复杂度和空间复杂度.详解

引言 详细介绍什么是时间复杂度和空间复杂度。 前言&#xff1a;为什么要学习时间复杂度和空间复杂度 算法在编写成可执行程序后&#xff0c;运行时需要耗费时间资源和空间(内存)资源。因此衡量一个算法的好坏&#xff0c;一般是从时间和空间两个维度来衡量的&#xff0c;即时…

Matlab 基于专家pid控制的时滞系统

1、内容简介 Matlab 185-基于专家pid控制的时滞系统 可以交流、咨询、答疑 2、内容说明 略 在处理时滞系统&#xff08;Time Delay Systems&#xff09;时&#xff0c;使用传统的PID控制可能会面临挑战&#xff0c;因为时滞会导致系统的不稳定或性能下降。专家PID控制通过结…

MyBatis源码分析のSql执行流程

文章目录 前言一、准备工作1.1、newExecutor 二、执行Sql2.1、getMappedStatement2.2、query 三、Cache装饰器的执行时机四、补充总结 前言 本篇主要介绍MyBatis解析配置文件完成后&#xff0c;执行sql的相关逻辑&#xff1a; public class Main {public static void main(Str…

【MySQL】数据库基础

目录 一、什么是数据库1.1 为什么要有数据库1.2 数据库的本质是什么1.3 在Linux下看一下数据库 二、主流数据库三、基本使用3.1 连接服务器3.2 服务器&#xff0c;数据库&#xff0c;表关系 四、MySQL架构五、SQL分类六、存储引擎6.1 存储引擎是什么6.2 查看存储引擎6.3 存储引…

算是解决可以访问github但无法clone的问题

本文的前提是使用了**且可以正常访问github 查看代理的端口 将其配置到git 首先查看git配置 git config --list然后添加配置&#xff0c;我这边使用的是Hiddfy默认的端口是12334&#xff0c;如果是clash应该是7890 git config --global http.proxy 127.0.0.1:12334其他 删除…

SpringBoot第三站:配置嵌入式服务器使用外置的Servlet容器

目录 1. 配置嵌入式服务器 1.1 如何定制和修改Servlet容器的相关配置 1.server.port8080 2. server.context-path/tx 3. server.tomcat.uri-encodingUTF-8 1.2 注册Servlet三大组件【Servlet&#xff0c;Filter&#xff0c;Listener】 1. servlet 2. filter 3. 监听器…

AdaLoRA 参数 配置:CAUSAL_LM“ 表示因果语言模型任务

AdaLoRA 参数 配置:CAUSAL_LM" 表示因果语言模型任务 config = AdaLoraConfig( init_r=16, # 增加 LoRA 矩阵的初始秩 lora_alpha=32, target_modules=[“q_proj”, “v_proj”], lora_dropout=0.1, bias=“none”, task_type=“CAUSAL_LM” ) 整体功能概述 AdaLoraCon…

IP 协议

文章目录 IP 协议概述数据包格式首部校验和实例分析实例一 分片抓包分析参考 本文为笔者学习以太网对网上资料归纳整理所做的笔记&#xff0c;文末均附有参考链接&#xff0c;如侵权&#xff0c;请联系删除。 IP 协议 概述 IP 协议是 TCP/IP 协议簇中的核心协议&#xff0c;也…

日常开发记录-radioGroup组件

日常开发记录-radioGroup组件 1.前提2.问题&#xff1a;无限循环调用3.解释Vue 事件传播机制分析与无限循环原因解释4.解决 1.前提 在上一章的&#xff0c;我们实现了radio组件。从这进入了解 新增个radioGroup组件呢。 <template><divclass"q-radio-group&quo…

API调用comfyui工作流,做一个自己的app,chatgpt给我写的前端,一键创建自己的卡通形象,附源码

前言 工具介绍 首先 comfyui你是少不了的&#xff0c;这个是工作流的后端支持&#xff0c;用这个去调试工作流和生成API可调用文件 前端我们就用很流行的gradio吧&#xff0c;什么你一时半会没有学gradio的计划&#xff0c;没事&#xff0c;笔者也没系统学过&#xff0c;我干…

【网络】数据流(Data Workflow)Routes(路由)、Controllers(控制器)、Models(模型) 和 Middleware(中间件)

在图片中&#xff0c;数据流&#xff08;Data Workflow&#xff09;描述了应用程序中数据的流动过程&#xff0c;涉及 Routes&#xff08;路由&#xff09;、Controllers&#xff08;控制器&#xff09;、Models&#xff08;模型&#xff09; 和 Middleware&#xff08;中间件&…