C++的左值,右值,左值引用,右值引用
参考大神链接: https://blog.csdn.net/u012198575/article/details/83142419
1.左值与右值
https://msdn.microsoft.com/en-us/library/f90831hc.aspx
左值(left-values),缩写:lvalues //也有locator value
右值(right-values),缩写:rvalues
所有的c++表达,不是左值就是右值。
lvalues是指存在于单个表达式之外的对象。你可以把左值当成有名字的对象。
所有的变量,包括常变量,都是左值。
rvalues是一个暂时存在的值存在于单个表达式之内的对象。
有点拗口(难理解),通俗来说就是,左值的生存期不只是这句话,后面还能用到它。
而右值呢,出了这句话就挂了,所以也叫(将亡值)。
简单定义:
lvalue代表一个在内存中占有确定位置的对象(换句话说就是有一个地址)。
rvalue通过排他性来定义,每个表达式不是lvalue就是rvalue。
因此从上面的lvalue的定义,rvalue是在不在内存中占有确定位置的表达式。
#include <iostream>
using namespace std;
int main()
{
int x = + ;
cout << x << endl;
}
x是左值,3 + 4是右值。
// lvalues_and_rvalues2.cpp
int main()
{
int i, j, *p; // 正确的使用: 变量是左值
i = ; // 错误的使用: 左边的操作 必须是 左值 (C2106)
= i; // C2106
j * = ; // C2106 // 正确的使用: 被间接引用的指针是左值
*p = i; const int ci = ;
// 错误的使用: 左边的操作 是 常量左值 (C3892)
ci = ; // C3892 // 正确的使用: 条件操作 返回了左值
((i < ) ? i : j) = ;
}
2.左值引用,右值引用
左值引用:https://msdn.microsoft.com/en-us/library/w7049scy.aspx
使用语法:类型 + &(引用符) + 表达式
可以把左值引用当成对象的另一个名字,lvalue引用声明由一个可选的说明符列表和一个引用声明符组成。
引用必须初始化,而且不能改变。
一个对象的地址可以 转化成 一种指定类型的指针 或者 转化成 一个 相似类型的引用。意义是相同的。
demo:
char c_val = 'c';
char *ptr = &c_val;//&c_val为取c_val地址
char &r_val = c_val;//左值引用表达式 r_val为c_val的引用
不要混淆 取地址 和 引用,当&说明符前面带有类型声明,则是引用,否则就是取地址。
通俗来说 &在"="号左边的是引用,
&在“=”右边的是取地址。
右值引用:https://msdn.microsoft.com/en-us/library/dd293668.aspx
使用语法:类型 + && + 表达式 //多了一个&符号
Move Semantics:移动语义
右值引用使您能够区分左值和右值。Lvalue引用和rvalue引用在语法和语义上是相似的。
右值引用支持移动语义的实现,可以显著提升应用程序的性能。移动语义允许您编写将资源(例如动态分配的内存)从一个对象传输到另一个对象的代码,移动语义行之有效,因为它允许从程序中其他地方无法引用的临时对象转移资源。
为了实现移动语义,你在类中提供一个动态构造,和可选择的动态赋值运算符(operator=)。拷贝和赋值操作的资源是右值的可以自动调用移动语义。不像缺省的拷贝构造,编译器并不提供缺省的动态构造。
#include <iostream>
#include <string>
using namespace std; int main()
{
string s = string("h") + "e" + "ll" + "o";
cout << s << endl;
}
在Visual C++ 2010之前,每个调用 “+”运算符会分配和返回一个新的临时的string对象,
“+”运算符不能从一个string扩展到另一个,因为它不知道string是左值还是右值。
如果源字符串都是lvalues,那么它们可能在程序的其他地方被引用,因此不能被修改。
通过使用右值引用“+”运算符能够修改那些不能在程序中别处引用的右值,所以现在“+”运算符可以有一个string扩展到另一个。
这可以显著减少字符串类必须执行的动态内存分配的数量。
为了更好地理解移动语义,考虑向向量对象插入一个元素的例子。
如果超出了vector对象的容量,vector对象必须为其元素重新分配内存,然后将每个元素复制到另一个内存位置,以便为插入的元素腾出空间。
当插入操作复制一个元素时,它创建一个新元素,调用copy构造函数将数据从前一个元素复制到新元素,然后销毁前一个元素。
移动语义允许您直接移动对象,而不必执行昂贵的内存分配和复制操作。
Perfect Forwarding:完美转发
完美的转发减少了重载函数 避免了转发的问题。转发的问题出现在你写通用函数将引用作为参数,将这些参数由函数调用的时候。
举个例子,如果通用函数将 type const T&作为参数,那么调用函数不能修改参数的值。
如果通用函数 将 type T&作为参数,那么当参数是右值的时候,函数不能调用。
通常来说,为了解决上述的问题,你需要提供重载函数,既要有type const T&参数的函数,也要有type T&参数的函数。
结果呢,重载函数的数量随着参数数量呈指数递增。而右值引用能够使你只用一个函数就能适用于任意数量的参数。
总结:
C++11引入了右值引用和移动语义。
右值引用将左值与右值区分开来。
它们可以帮助您通过消除不必要的内存分配和复制操作来提高应用程序的性能。
它们还使您能够编写接受任意参数的函数的一个版本,并将其转发给另一个函数,就好像直接调用了另一个函数一样
C++的左值,右值,左值引用,右值引用的更多相关文章
- ecos3.0编译 if_lancepci.c:528: 错误: 赋值运算的左操作数必须是左值
/home/xin/ecos3/ecos-3.0/packages/devs/eth/amd/lancepci/v3_0/src/if_lancepci.c:528: 错误: 赋值运算的左操作数必须是 ...
- item 24: 区分右值引用和universal引用
本文翻译自<effective modern C++>,由于水平有限,故无法保证翻译完全正确,欢迎指出错误.谢谢! 博客已经迁移到这里啦 古人曾说事情的真相会让你觉得很自在,但是在适当的情 ...
- [android]如何使LinearLayout布局从右向左水平排列,而不是从左向右排列
方法1: <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:l ...
- 5-2-2 printf参数从右往左压栈
5-2-2 C中printf计算参数时是从右到左压栈的 #include <stdio.h> int main (int argc, char **argv) { ; ,,,,}; int ...
- Ueditor增加文字竖排显示和从右向左输入(支持蒙古文和维吾尔文)
平时我们在ueditor中都是输入的中文,排版都是从左向右输入.但是当输入一些少数民民族言时,ueditor却不能正常显示. 例如蒙古文字,传统蒙文是竖排书写.如下图:传统蒙古文排文方式,<蒙古 ...
- const引用返回值
一.引用 引用是别名 必须在定义引用时进行初始化.初始化是指明引用指向哪个对象的唯一方法. const 引用是指向 const 对象的引用: ; const int &refVal = iva ...
- C语言函数入参压栈顺序为什么是从右向左?
看到有人提问到,在处理printf/cout时,压栈顺序是什么样的?大家都知道是从右往左,也就是说从右往左的计算,但是,这里的计算不等于输出. a++和++a的压栈的区别:在计算时,遇到a++会记录此 ...
- C/C++---printf/cout 从右至左压栈顺序实例详解
__cdecl压栈顺序实例 明白计算:计算是从右到左计算的 栈和寄存器变量:x++,是将计算结果存放到栈空间,最后是要出栈的:而++x和x是将计算结果直接存放到某个寄存器变量中(是同一个),所以计算完 ...
- i++为什么不能作为左值,而++i可以作为左值
今天看书见到如下代码: int a=2; ++a++; 根据操作符的优先级和结合性知,操作符++的优先级为3,结合性为右结合,即++a++;可以理解为++(a++); 但我把代码放在vs2015上,结 ...
- Python读字节某一位的值,设置某一位的值,二进制位操作
Python读字节某一位的值,设置某一位的值,二进制位操作 在物联网实际应用项目开发中,为了提升性能,与设备端配合,往往最终使用的是二进制字节串方式进行的通信协议封装,更会把0和1.True和Fa ...
随机推荐
- Lucas定理初探
1.1 问题引入 已知\(p\)是一质数,求\(\dbinom{n}{m}\pmod{p}\). 关于组合数,它和排列数都是组合数学中的重要概念.这里会张贴有关这两个数的部分内容. 由于Lucas定理 ...
- [LOJ3109][TJOI2019]甲苯先生的线段树:DP
分析 首先,请允许我 orz HN队长zsy.链接 我们发现树上的链有两种类,一类是直上直下的,一类不是直上直下的(废话).并且,如果我们确定了左侧和右侧的链的长度和整条链上所有节点的编号之和,那么这 ...
- ubuntu16.04配置记录
新开一篇随笔记录ubuntu16.04配置中遇到的坑 1.安装Bumblebee(大黄蜂) Bumblebee是一款双显卡驱动,可以关闭独显,有效控制笔记本发热 第一步:安装我们的主角Bumblebe ...
- 17.Python print()函数高级用法
前面使用 print() 函数时,都只输出了一个变量,但实际上 print() 函数完全可以同时输出多个变量,而且它具有更多丰富的功能. print() 函数的详细语法格式如下: print (val ...
- Link Script 学习
最后更新 2019-06-27 概述 当使用 C 或者 C++ 编写代码实现某种功能时,需要将源代码进行编译以及链接.链接是将一系列目标文件(.o)以及归档文件(.a)组合起来,重新定位各个文件数据并 ...
- XStream详解
XStream的作用 XStream可以把JavaBean对象转换成XML! 通常服务器向客户端响应的数据都是来自数据库的一组对象,而我们不能直接把对象响应给客户端,所以我们需要把对象转换成XML再响 ...
- ES6学习一--ES5
JSON对象方法 1.将JSON字符串转换成js对象(IE7及以下不支持)(IE7及以下兼容需其他手段) JSON.parse() 2.将js对象转成JSON字符串 JSON.stringify() ...
- MYSQL中唯一约束和唯一索引的区别
1.唯一约束和唯一索引,都可以实现列数据的唯一,列值可以有null.2.创建唯一约束,会自动创建一个同名的唯一索引,该索引不能单独删除,删除约束会自动删除索引.唯一约束是通过唯一索引来实现数据的唯一. ...
- Ruby Programming学习笔记
#将ARGV[0]转换成正则表达式类型 pattern= Regexp.new(ARGV[0]) #Devise gem包 Devise是Ruby中使用最广泛的身份验证gem包之一.Devise为我们 ...
- openssl-1.0.1u静态库编译
不管Windows还是linux都是需要安装好perl环境的 Windows步骤 1.解压openssl-1.0.1u.tar.gz 2.使用Vs2005命令行工具进入解压后的目录 3.执行如下命令 ...