【C++】C++11--右值引用

news/2025/1/16 0:47:54/

文章目录

  • C++11 --- 右值引用
    • 右值 and 左值
    • 左值引用 and 右值引用
      • 右值引用的使用场景和意义
      • 右值引用和移动语义
      • 右值引用左值
    • 完美转发
      • 万能引用
      • 完美转发保持属性

C++11 — 右值引用

右值 and 左值

左值是什么

左值是一个表示数据的表达式,比如变量名和可以解引用的指针

  • 左值可以出现在赋值符号的两边

  • 左值可以被取地址,也可以被修改(const修饰的左值除外)

右值是什么

右值也是一个表达数据的表达式,如常量,表达式的返回值,函数的返回值等等

  • 右值不可以被取地址,也不可以被修改
  • 右值可以出现在赋值符号的右边,到那时不能出现在赋值符号的左边
int x = 1, y = 2;
// 以下几个表达式都是常见的右值
10;
x + y;
min(x, y);
  • 右值本质就是一个临时变量或者是常量,10就是常量而x + y,min(x, y)的返回值都是临时变量,这种无法被更改的值我们称之为右值
  • 这些临时变量和常量目前并未被存储起来,也就无法取地址
  • 需要注意,传值返回的函数的返回值才是右值,因为传值返回的函数返回的是对象的拷贝,这个拷贝出来的对象是一个临时变量

对于左值引用返回的函数来说,返回值是左值。比如unordered_map[]运算符重载其返回的就是kv中的value的引用,我们可以对其进行赋值

左值引用 and 右值引用

C++11中新增了右值引用的语法特性,但是不论是左值引用还是右值引用,本质都是给对象去别名

左值引用

左值引用就是给左值去别名,通过&来声明

int	a = 10; 								int& ra = a;
int* p = new int(10);  			int*& rp = p;
const int c = 2; 						const int& rc = c;

右值引用

右值引用解释给右值取别名,通常使用&&来声明

int x = 1, y = 2;int&& rr1 = 10;
int&& rr2 = x + y;

需要注意:右值是不可以取地址的,但是给右值取别名后,会导致右值被存储到特定位置,这个时候到右值可以被取到地址,并且可以被修改,如果不想让被引用的右值被修改,可以使用const修饰右值引用

const 左值引用 and const 右值引用

void test5() {const int&& ra = 10;int *p = const_cast<int*>(&ra);*p = 2;cout << "&ra = " << &ra << endl;cout << "p = " << p << endl;cout << "ra = " << ra << endl;cout << "*p = " << *p << endl;cout << endl;const int x = 10;const int& b = x;int *pb = const_cast<int*>(&b);*pb = 5;cout << "&b = " << &b << endl;cout << "pb = " << pb << endl;cout << "b = " << b << endl;cout << "*pb = " << *pb << endl;
}
&ra = 0x30445b464
p = 0x30445b464
ra = 2
*p = 2&b = 0x30445b454
pb = 0x30445b454
b = 5
*pb = 5

可以看到左值引用或是右值引用它们都是一样的,他们都在内存上开辟了空间并将数据存到了空间中,当我们访问空间时就会将空间中的值返回给我们。

注意:左值引用的const 变量是不会被写入常量表的,也不会进行宏替换。其会保持内存可见性访问该变量时会到内存中获取该变量的值

Const 左值引用右值

左值引用不可以直接引用右值,因为这涉及到权限的放大,右值不可被修改但是左值引用可以修改。但是const修饰的左值引用可以引用右值,因为const左值引用能搞保证被引用的数据不被修改

void test6() {const int& c = 10;int* pc = const_cast<int*>(&c);*pc = 5;cout << "c = " << c << endl;     // 5cout << "*pc = " << *pc << endl; // 5
}

可以看到不管是左值引用还是右值引用亦或是const左值引用右值都不会被写入常量表

右值引用move左值

右值引用只能引用右值,不能引用左值。但是右值引用可以引用move以后的左值。move函数是C++标准提供的函数,被move后的左值能够赋值给右值引用

void test7() {int a = 10;int&& b = std::move(a);  b = 3;	cout << a << endl;				 // 输出3
}

右值引用的使用场景和意义

虽然const左值引用既能够接收左值,也可以接收右值,但是左值引用终究存在短板,而C++11提出的右值引用就是来解决左值引用的短板的

准备工作

这里我们使用了前面模拟实现STL容器中的string类。类中实现了一些基本函数

//
// Created by 陈李鑫 on 2023/7/16.
//
#ifndef SIMULATION_REALIZATION_STL_CLX_STRING_HPP
#define SIMULATION_REALIZATION_STL_CLX_STRING_HPP#endif //SIMULATION_REALIZATION_STL_CLX_STRING_HPP#include <iostream>
#include <algorithm>
#include <utility>
#include <cassert>class clx_string{
public:typedef char* iterator;iterator begin() { return _str;}iterator end() { return _str + _size; }const char* c_str() const { return const_cast<const char*>(_str); };void swap(clx_string& s);clx_string(const char* str = "");clx_string(const clx_string& s);~clx_string();clx_string& operator=(const clx_string& s);char& operator[](size_t i);void reserve(size_t n);void push_back(char ch);clx_string& operator+=(char ch);
private:char* _str;size_t _size;size_t _capacity;
};void clx_string::swap(clx_string& s) {std::swap(_size, s._size);std::swap(_capacity, s._capacity);std::swap(_str, s._str);
}clx_string::clx_string(const char* str) {std::cout << "clx_string(const char* str) -- 直接构造" << std::endl;_size = strlen(str);_capacity = _size;_str = new char[_capacity + 1];strcpy(_str, str);
}// 拷贝构造函数 以前的写法
// clx_string::clx_string(const clx_string& s) {
//     _size = strlen(s.c_str());
//     _capacity = _size;
//     _str = new char[_capacity + 1];
//     strcpy(_str, s.c_str());
// }// 拷贝构造函数 现代写法
clx_string::clx_string(const clx_string& s): _str(nullptr), _size(0), _capacity(0)
{std::cout << "clx_string(const clx_string& s) -- 拷贝构造"  << std::endl;clx_string tmp(s.c_str());swap(tmp);std::cout << std::endl;std::cout << std::endl;
}clx_string::~clx_string() {_size = 0;_capacity = 0;delete[] _str;_str = nullptr;
}
clx_string& clx_string:: operator=(const clx_string& s) {std::cout << "clx_string& clx_string:: operator=(const clx_string& s) -- 赋值函数重载" << std::endl;clx_string tmp(s.c_str());clx_string::swap(tmp);std::cout << std::endl;std::cout << std::endl;return *this;
}
char& clx_string::operator[](size_t i) {assert(0 <= i && i < _size);return _str[i];
}void clx_string::reserve(size_t n) {if (n > _capacity) {char* tmp = new char[n + 1];strncpy(tmp, _str, _size + 1);delete[] _str;_str = tmp;_capacity = n;}
}
void clx_string::push_back(char ch) {while (_size >= _capacity) {reserve(_capacity == 0 ? 4 : _capacity * 2);}_str[_size] = ch;_str[_size + 1] = 0;_size++;
}
clx_string& clx_string::operator+=(char ch) {push_back(ch);return *this;
}

这里主要关注两个函数,一是拷贝构造函数,二是赋值运算符重载,它们内部都包含了一个构造函数

 clx_string tmp(s.c_str());
// 拷贝构造函数 现代写法
clx_string::clx_string(const clx_string& s): _str(nullptr), _size(0), _capacity(0)
{std::cout << "clx_string(const clx_string& s) -- 拷贝构造"  << std::endl;clx_string tmp(s.c_str());swap(tmp);std::cout << std::endl;std::cout << std::endl;
}clx_string& clx_string:: operator=(const clx_string& s) {std::cout << "clx_string& clx_string:: operator=(const clx_string& s) -- 赋值函数重载" << std::endl;clx_string tmp(s.c_str());clx_string::swap(tmp);std::cout << std::endl;std::cout << std::endl;return *this;
}

我们可以写一个简单的案例测试一下每个函数的调用打印是否清晰

void clx_string_test1() {clx_string s1;cout << endl;clx_string s2(s1);clx_string s3;s3 = s1;
}
clx_string(const char* str) -- 直接构造        // s1 的直接构建clx_string(const clx_string& s) -- 拷贝构造		 // s2 的拷贝构建
clx_string(const char* str) -- 直接构造clx_string(const char* str) -- 直接构造        // s3 的直接构建后调用赋值函数重载 
clx_string& clx_string:: operator=(const clx_string& s) -- 赋值函数重载
clx_string(const char* str) -- 直接构造

这样我们的准备工作就算完成了,接下来就要进入右值引用的价值的正是讲解

左值引用的使用场景

在介绍左值引用的短板之前,我们先强调一下左值引用的价值

  • 左值引用做参数,防止传参时进行拷贝
  • 左值引用做返回值,防止返回时对返回对象进行拷贝
clx_string func1(clx_string s) { return s; }
clx_string& func2(clx_string& s) { return s; };void clx_string_test2() {clx_string s1;std::cout << std::endl;std::cout << "func1 begin" << std::endl;func1(s1);std::cout << "func1 end" << std::endl;std::cout << std::endl;std::cout << "func1 begin" << std::endl;func2(s1);std::cout << "func1 end" << std::endl;
}
clx_string(const char* str) -- 直接构造func1 begin
clx_string(const clx_string& s) -- 拷贝构造
clx_string(const char* str) -- 直接构造clx_string(const clx_string& s) -- 拷贝构造
clx_string(const char* str) -- 直接构造func1 endfunc1 begin
func1 end

可以看到使用引用传参和引用返回减少了两次的拷贝构造,我们知道stirng这种类型进行的拷贝都是深拷贝,如果string很大那么深拷贝的代价是非常高的,使用左值传参和做返回值起到的作用还是非常明显的

左值引用的短板

左值引用虽然在某些情况下可以避免不必要的拷贝操作,但是并不能完全避免。

  • 函数返回的对象是一个局部变量,该变量出了函数作用域就被销毁了,这种情况不能使用左值引用作为返回值,只能以传值的方式返回,这就是左值引用的短板(左值引用使用的前提是出了这个域,左值任然存在)
clx_string clx_string::to_string(int value) {clx_string res;bool flag = false;if (value < 0) {flag = true;value = -1 * value;}while (value > 0) {char ch = static_cast<char>(value % 10);res += ch + '0';value /= 10;}if(flag) res += '-';std::reverse(res.begin(), res.end());return res;
}
void clx_string_test3() {clx_string s;s = clx_string::to_string(1);
}
clx_string(const char* str) -- 直接构造  // 直接构造s
clx_string(const char* str) -- 直接构造  // to_string 内部构建 res
clx_string& clx_string:: operator=(const clx_string& s) -- 赋值函数重载    // s接收返回值
clx_string(const char* str) -- 直接构造

比如to_string函数就不可能使用左值返回,因为该函数生成的字符串就在函数域中,出了函数域就销毁了,因此该函数只能返回一个局部变量所以必须传值返回

当传值返回后我们需要调用变量来接收to_string函数返回的局部变量这里又要调一次拷贝函数,那么to_stirng函数在函数内生成字符串(第一次),传值返回(第二次),将其的数据传递给接收变量(第三次)可以看到同一份数据生成了三次,也就是说在这之间白白进行了两次深拷贝

注意:近代编译器对上述情况进行了优化,使得传值返回的变量可以直接给父域的对象拷贝构造,可以减少一次拷贝

C++11提出右值引用就是为了解决左值引用这个短板的,但解决方式并不是简单的将右值引用作为函数的返回值

右值引用和移动语义

右值引用和移动语义解决上述问题的方法就是,给当前模拟实现的string类增加移动构造和移动赋值的方法

移动构造

移动构造是一个构造函数,该构造函数的参数就是右值类型的,移动构造的本质就是将传入的右值资源窃取过来,占为己有,这样就避免了深拷贝

在当前string类中新增一个移动构造函数,该函数要做的就是调用swap函数将传入的右值的资源窃取过来

clx_string::clx_string(clx_string&& s):_size(0), _capacity(0), _str(nullptr)
{std::cout << "clx_string::clx_string(clx_string&& s) -- 移动构造" << std::endl;swap(s);
}clx_string& clx_string::operator=(clx_string&& s) {std::cout << "clx_string& clx_string::operator=(clx_string&& s) -- 移动赋值重载" << std::endl;swap(s);return *this;
}

移动构造和拷贝构造的区别:

  • 拷贝构造采用的一直是const 左值引用接收参数,因此无论拷贝构造对象传入的是左值还是右值,都会调用拷贝构造函数

  • 增加移动构造之后,如果传入的参数是右值,那么就会匹配到移动构造函数(最匹配原则)

  • string的拷贝构造函数做的就是深拷贝,而移动构造函数只需要调用swap函数进行资源转移,移动构造的代价比拷贝构造小很多

void clx_string_test3() {clx_string s1;s1 = clx_string::to_string(1);
}
clx_string(const char* str) -- 直接构造				// to_string 	创建res
clx_string(const char* str) -- 直接构造  			// 创建 s1
clx_string& clx_string::operator=(clx_string&& s) -- 移动赋值重载  // 返回值移动赋值给s1

可以看到又了移动构造和移动拷贝,就不会调用原来的拷贝创建以及拷贝赋值了,使用了移动构造替换拷贝构造,移动赋值替换拷贝赋值,提高了效率

注意⚠️:虽然to_string中的局部string 对象是一个左值,但由于该string对象在当前函数调用结束后就会被立即销毁,我们可以把这种被消耗的值叫做将亡值,比如匿名对象也可以叫做将亡值。即然将亡值都要被销毁了,还不如把自己的资源转移给别人,因此编译器会讲这种将亡值识别为右值,这样就可以匹配到参数类别为右值的移动构造函数

STL中的容器

C++11标准出来后,STL容器都增加了移动构造和移动赋值,以我们刚刚说的string为例

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-gF0dvrSO-1689499058827)(/Users/chenlixin/Library/Application Support/typora-user-images/image-20230716141556485.png)]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-rByFUe8c-1689499058828)(/Users/chenlixin/Library/Application Support/typora-user-images/image-20230716141712003.png)]

右值引用还在各种容器的插入中使用

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-2NFd0hPw-1689499058828)(/Users/chenlixin/Library/Application Support/typora-user-images/image-20230716144526460.png)]

C++11后很多STL容器的push_back类似接口都提供了右值引用版本,如果传入的参数是右值可以直接进行资源转移,避免了深拷贝,提高了效率

右值引用左值

右值引用虽然不能直接引用左值,但是可以通过move函数将左值转化成右值。move函数的名称非常具有迷惑性,move函数其实并不能搬移任何东西,该函数的唯一功能就是将一个左值强转成右值引用,然后实现移动语义

// 声明
template <class T>
typename remove_reference<T>::type&& move (T&& arg) noexcept;//实现
template <class T>
typename remove_reference<T>::type&& move (T&& arg) noexcept {return ((typename remove_reference<T>::type&&)_Arg);
}

move函数中arg参数类型并非右值引用,而是万能引用。万能引用跟右值引用的形式相同,但是右值引用必须得钥匙确定类型的。一个左值被move后,它的资源有可能已经被转移给别人了,因此要慎用一个被move后的左值

void clx_string_test4() {clx_string s1("hello world");clx_string s2;s2 = std::move(s1);cout << "s1 : " << s1.c_str() << endl;cout << "s2 : " << s2.c_str() << endl;
}int main() {clx_string_test4();return 0;
}
clx_string(const char* str) -- 直接构造
clx_string(const char* str) -- 直接构造
clx_string& clx_string::operator=(clx_string&& s) -- 移动赋值重载
s1 : 
s2 : hello world

可以看到s1本来是一个左值,我们将其强转成右值赋值给了s2,那么s1内部的资源就已经被转走了,所以一个左值被move后资源可能被转移其是很危险的,慎用

完美转发

万能引用

模版中的&&不能代表右值引用,而是万能引用,其既能接受左值也能接受右值。万能引用和右值引用的区别就是,右值引用必须要确定类型,而万能引用是根据传入实参的类型进行推导

void Func(int& x) {cout << "左值引用" << endl;
}
void Func(const int& x) {cout << "const 修饰的左值引用" << endl;
}
void Func(int&& x)  {cout << "右值引用" << endl;
}
void Func(const int&& x) {cout << "const 修饰的右值引用" << endl;
}
template<class T>
void PerfectForward(T&& t) {Func(t);
};void clx_string_test5() {int a = 10;PerfectForward(a);          // 左值PerfectForward(move(a));    // 右值const int b = 10;PerfectForward(b);					    // const修饰的左值PerfectForward(std::move(b));   // const修饰的右值
}
左值引用
左值引用
const 修饰的左值引用
const 修饰的左值引用

由于PerfectForward函数的参数类型是万能引用,因此既可以接受左值也可以接受右值,我们在PerfectForward中调用Func函数,希望我们传什么类型的值就能给我们匹配什么类型的函数

  • 但是我们实际调用发现,无论我们传左值还是右值,匹配到的全部都是左值版本的Func函数,这时因为右值引用后对导致右值被存储到特定位置,这时候的右值可以被取到地址也可以被修改,会被识别成左值

完美转发保持属性

如果想要在参数的传递过程中保持其原有的属性,就需要在传参的时候调用forward函数

template<class T>
void PerfectForward(T&& t) {Func(std::forward<T>(t));
};

经过完美转发后,调用PerfectForward函数传入的右值就会被保持右值属性,就会匹配到右值版本,这就是完美转发的价值


http://www.ppmy.cn/news/911701.html

相关文章

cesium的使用

cesium的使用 cesium的使用创建一个vue项目 vuevitecesium参数的使用常用点位标记删除动态渲染路线借助truf.js的算法进行渲染地块的实现topojson cesium的使用 1.下载或者安装cesium的插件 官方文档 下载下来后创建文件夹整个包引入 2.生成token 新的包应该有默认token如果没…

俄罗斯方块,消消乐源码下载

这些小游戏是java做的&#xff0c;参考源码&#xff0c;可以用C、安卓实现。 按照复杂程度区分版本&#xff0c;方便初学者学习。做的时候很匆忙&#xff0c;是为了教学演示。如果要改进&#xff0c;还是用设计模式、解耦、状态和标志&#xff0c;仍然是基于游戏的MVC结构&…

符号消消乐

最近看到的一道机试题&#xff0c;输入一个只包含英文字母的字符串&#xff0c;字符串中的俩个字母如果相邻且相同&#xff0c;就可以消除。在字符串上反复执行消除的动作&#xff0c;直到无法继续消除为止&#xff0c;此时游戏结束。输出最终得到的字符串长度 package com.cl…

开心消消乐在屏幕中向下移动的完整代码

#--coding:utf8-- import random import pygame as pgWIDTH 400 HEIGHT 400 NUMGRID 8 GRIDSIZE 36 XMARGIN (WIDTH - GRIDSIZE * NUMGRID) // 2 YMARGIN (HEIGHT - GRIDSIZE * NUMGRID) // 2# 加载图片 gem_imgs [pg.image.load(resources/images/gem{}.png.format(i))…

消除类游戏

数组 ** xdoj-消除类游戏** #include<stdio.h> int main() {int n, m, i, j;scanf("%d%d",&n,&m);int a[30][30], b[30][30];for(i0;i<n;i){for(j0;j<m;j){scanf("%d",&a[i][j]);b[i][j] a[i][j];}}for(i0;i<n;i){for(j0;j…

Python学习:方块消除游戏

【编程题&#xff1a;方块消除游戏】 emmmm..依然牛客网上的题&#xff0c;难度标识才一颗星&#xff0c;但是感觉自己看代码看了好久才理解实现的过程。 题目描述&#xff1a;如下图&#xff0c;有10*10个不同颜色的方块&#xff0c;每个方块可能是红、绿、蓝、黄、紫5种颜色…

【AI】微信小程序 方块消消消

介绍 大概是下图所述一款游戏…… 感觉这种能(创造一种新的玩法)的游戏超级厉害 (最高分了……并不擅长这种游戏) 然后当初玩的时候非常的上瘾……主要是每次凉的时候都有一种(明明还空很多只是时运不齐命途多舛所以这局死那么早)的错觉。 为了显得我没有浪费时间………

[Cocos Creator] 制作简版消消乐(三):实现方块的生成与交换

本文由“壹伴编辑器”提供技术支 前言 在上一篇文章中我们实现部分基础组件和管理脚本&#xff0c;那么本篇文章将和大家一起实现方块的生成与交换的逻辑。 温馨提醒&#xff1a;本文含有大量代码和注释&#xff0c;请提前做好心理准备并认真阅读。 话不多说&#xff0c;冲鸭&a…