C++ 右值引用

1.左值引用和右值引用的概念

什么是左值?什么是左值引用?

左值是一个表示数据的表达式(如变量名或解引用的指针),我们可以获取它的地址+可以对它赋值,左值可以出现赋值符号的左边,右值不能出现在赋值符号左边。定义时const修饰符后的左值,不能给他赋值,但是可以取它的地址。左值引用就是给左值的引用,给左值取别名。
什么是右值?什么是右值引用?

右值也是一个表示数据的表达式,如:字面常量、表达式返回值,函数返回值(这个不能是左值引用返回)等等,右值可以出现在赋值符号的右边,但是不能出现出现在赋值符号的左边,右值不能取地址。右值引用就是对右值的引用,给右值取别名。

2.左值引用与右值引用比较

左值引用总结:

  1. 左值引用只能引用左值,不能引用右值。

  2. 但是const左值引用既可引用左值,也可引用右值。
    右值引用总结:

  3. 右值引用只能右值,不能引用左值。

  4. 但是右值引用可以move以后的左值。

cpp 复制代码
	int b = 10;
	int&& a = 10;
	int&& c = move(b);

3.为什么引入右值引用

左值引用的短板:

但是当函数返回对象是一个局部变量,出了函数作用域就不存在了,就不能使用左值引用返回,传值返回会导致至少1次拷贝构造(如果是一些旧一点的编译器可能是两次拷贝构造)。
右值引用和移动语义解决上述问题:

移动构造,就是窃取别人的资源来构造自己。试想一下如果是数据量庞大的自定义类型,能够尽量减少拷贝,将会提高语言效率。

cpp 复制代码
//移动构造
xxx(xxx&& s)
{
	cout << "xxx(xxx&& s) -- 移动语义" << endl;
	swap(s);
    //例如直接交换指针的指向,因为右值一般生命周期快要结束了
    //就像武林高手临终前传授功力给主角,主角就减少了修炼成本
}
//移动赋值
xxx& operator=(xxx&& s)
{
	cout << "xxx& operator=(xxx&& s) -- 移动语义" << endl;
	swap(s);
	return *this;
}

4.完美转发

模板中的&&不代表右值引用,而是万能引用,其既能接收左值又能接收右值。

模板的万能引用只是提供了能够接收同时接收左值引用和右值引用的能力,

但是引用类型的唯一作用就是限制了接收的类型,后续使用中都退化成了左值,

我们希望能够在传递过程中保持它的左值或者右值的属性, 就需要用我们下面学习的完美转发。

完美转发前:

完美转发后:

相关推荐
无敌最俊朗@21 小时前
STL-deque面试剖析(面试复习4)
开发语言
APIshop21 小时前
用 Python 把“API 接口”当数据源——从找口子到落库的全流程实战
开发语言·python
Java Fans21 小时前
Qt Designer 和 PyQt 开发教程
开发语言·qt·pyqt
RwTo21 小时前
【源码】-Java线程池ThreadPool
java·开发语言
兮动人21 小时前
EMT4J定制规则版:Java 8→17迁移兼容性检测与规则优化实战
java·开发语言·emt4j
一点★21 小时前
Java中的常量池和字符串常量池
java·开发语言
咬人喵喵21 小时前
14 类圣诞核心 SVG 交互方案拆解(附案例 + 资源)
开发语言·前端·javascript
SunkingYang21 小时前
程序崩溃闪退——MFC共享内存多次OpenFileMapping和MapViewOfFile而没有相应的UnmapViewOfFile和CloseHandle
c++·mfc·共享内存·崩溃·闪退·close·openfilemapping
开始了码1 天前
深入理解回调函数:从概念到 Qt 实战
开发语言·qt
问君能有几多愁~1 天前
C++ 日志实现
java·前端·c++