概述

移动

移动(move)语义C++引入了一种新的内存优化,以避免不必要的拷贝。在构造或者赋值的时候,如果实参是右值(或者左值由std::move转换成右值),便会匹配移动语义的函数调用如下述举例的Str(Str&& obj)

移动语义的本质是将资源(内存/句柄)转移给另一个对象,被转移资源的对象不应再被使用。(这个概念有点像仙侠小说中的夺舍,夺舍成功的人获取被夺舍的人的身体(资源)),如下面伪代码:

class Obj
{
data
Obj(){
data = malloc(100)
}
// 移动 (夺舍)
Obj(Obj&& other){
data = other.data
other.data = nullptr
}
}

右值

右值直观理解是等号右边的值(大概如此,并不准确),右值的概念指代的东西比较多,大概是指不可寻址的值(也有例外)。我觉得这个不必太过纠结,记住几个常见的即可:

  • 临时对象:如函数返回的临时对象(下面有举例)
  • 字面量
  • 显式std::move()转换的值
  • 没有捕获参数的lambda

C++ 值类别表

在 C++11之后,C++根据

  • 被标识:可通过不同标识符指代同一实体。(对象/内存)
  • 可移动:可作为移动语义函数的参数,例如移动构造,移动赋值。

将值分为以下类别:

  • 泛左值:被标识

    • 左值:被标识且不可移动
    • 将亡值:被标识可移动
  • 右值:可移动
    • 将亡值:被标识可移动
    • 纯右值:不被标识且可移动

用法举例

参考测试项目代码ModernCppTest/modrenc_rvalueref_stdmove.cpp

主要内容:

  • 移动语义下的构造和赋值
  • 移动还是拷贝的重载匹配
  • C++ 优化临时对象(连加产生的中间临时对象)尝试调用移动语义
#include "ModernCppTestHeader.h"
#include <string>
using std::string; namespace n_rvalueref {
class Str {
public:
Str() {
LOG("无参构造");
this->str = new string();
} Str(const string& str) {
LOG("有参构造 str = " << str);
this->str = new string(str);
} Str(const Str& obj) {
LOG("拷贝构造 obj.str = " << *obj.str);
this->str = new string(*obj.str);
} Str(Str&& obj) noexcept {
LOG("移动构造 obj.str = " << *obj.str);
this->str = std::move(obj.str);
// 被移动的对象不应该再被使用了
obj.str = nullptr;
} Str& operator=(Str&& v) noexcept {
LOG("移动语义 operator = "); if (this != &v) {
this->str = std::move(v.str);
} return *this;
} Str operator+(const Str& v)
{
string s = *this->str + *v.str;
return Str(s);
} void Log()
{
LOG(str);
} string* str;
};
} using n_rvalueref::Str; // 右值引用&移动语义
void rvalueref_stdmove_test()
{
LOG_FUNC(); LOG_TAG("拷贝构造");
{
Str t1("A");
Str t2 = t1;
LOG_VAR(*t2.str);
} LOG_TAG("移动构造, 注意被移动的对象t1不应再被使用");
{
// t1是左值,使用std::move强制转换成右值
Str t1("A");
Str t2 = std::move(t1);
LOG_VAR(*t2.str);
} LOG_TAG("移动语义的运算符重载,注意运算符重载发生赋值运算(这个例子),而不是构造运算(上个例子)");
{
Str t1("A");
Str t2;
t2 = std::move(t1);
} LOG_TAG("除了上述显示使用std::move转换,常见的容易忽视的发生移动构造场合列举");
{
LOG("---1 连续加法产生的临时对象,c++会尝试使用移动语义进行优化");
Str t1("A");
Str t2("B");
Str t3("C");
Str t4;
t4 = t1 + t2 + t3; LOG("---2 函数返回的临时对象,c++会尝试使用移动语义进行优化");
auto f = []() {
auto s = Str("Hi");
return s;
};
Str t5 = f(); /*
- 在容器中插入或删除元素:比如 std::vector::push_back,如果传递给它的是右值,它就会使用移动语义。
- 在标准库算法中:许多标准库算法,比如 std::sort,std::partition 等,在进行元素交换时会使用移动语义。
- 在 std::swap 中:std::swap 会使用移动语义来交换两个对象。
*/
}
}

现代C++(Modern C++)基本用法实践:三、移动语义的更多相关文章

  1. nodejs 实践:express 最佳实践(三) express 解析

    nodejs 实践:express 最佳实践(三) express 解析 nodejs 发展很快,从 npm 上面的包托管数量就可以看出来.不过从另一方面来看,也是反映了 nodejs 的基础不稳固, ...

  2. Linux中sed的用法实践

    Linux中sed的用法实践 参考资料:https://www.cnblogs.com/emanlee/archive/2013/09/07/3307642.html http://www.fn139 ...

  3. WebSocket原理与实践(三)--解析数据帧

    WebSocket原理与实践(三)--解析数据帧 1-1 理解数据帧的含义:   在WebSocket协议中,数据是通过帧序列来传输的.为了数据安全原因,客户端必须掩码(mask)它发送到服务器的所有 ...

  4. Linux及安全实践三——程序破解

    Linux及安全实践三--程序破解 一.基本知识 常用指令机器码 指令 作用 机器码 nop 无作用(no operation) 90 call 调用子程序,子程序以ret结尾 e8 ret 返回程序 ...

  5. jquery jtemplates.js模板渲染引擎的详细用法第三篇

    jquery jtemplates.js模板渲染引擎的详细用法第三篇 <span style="font-family:Microsoft YaHei;font-size:14px;& ...

  6. Linux find命令的用法实践

    一.find命令简介 Linux下find命令在目录结构中搜索文件,并执行指定的操作.Linux下find命令提供了相当多的查找条件,功能很强大.由于find具有强大的功能,所以它的选项也很多,其中大 ...

  7. AI (Adobe Illustrator)详细用法(三)

    本文主要是介绍和色彩相关的用法. 一.路径外观设置 1.设置描边粗细 手动输入20px 下拉选择 鼠标选中数值,按向上或向下的箭头调整 在右边的描边菜单中修改 Note:按住shift键,然后上下箭头 ...

  8. Android最佳性能实践(三)——高性能编码优化

    在前两篇文章当中,我们主要学习了Android内存方面的相关知识,包括如何合理地使用内存,以及当发生内存泄露时如何定位出问题的原因.那么关于内存的知识就讨论到这里,今天开始我们将学习一些性能编码优化的 ...

  9. 并发编程实践三:Condition

    Condition实例始终被绑定到一个锁(Lock)上.Lock替代了Java的synchronized方法,而Condition则替代了Object的监视器方法,包含wait.notify和noti ...

  10. ansible playbook实践(三)-yaml文件写法

    playbook基于YAML语法来编写,基本语法规则如下: 1.大小写敏感 2.使用缩进表示层级关系 3.缩进时不允许使用Tab键,只允许使用空格 4.缩进的空格数目不重要,只要相同层级的元素左侧对齐 ...

随机推荐

  1. 如何建设一个用于编译 iOS App 的 macOS 云服务器集群?

    作者:京东零售 叶萌 现代软件开发一般会借助 CI/CD 来提升代码质量.加快发版速度.自动化重复的事情,iOS App 只能在 mac 机器上编译,CI/CD 工具因此需要有一个 macOS 云服务 ...

  2. UniApp小程序开发项目创建与运行

    1.准备工作:HbuiderX  +  微信开发者工具下载安装+小程序账号申请开通(这里就不例举了,可以看同账号uniapp小程序开发准备) 2.创建项目 新版本的HbuilderX点击新建项目--选 ...

  3. MQTT-主题基础

    MQTT主题 MQTT的主题是一个utf-8编码的字符串,最大长度65535字节,严格区分大小写 MQTT主题支持分层结构,主题分隔符用'/'表示,主题的层级长度可以为0 # 将主题划分为3个层级 ' ...

  4. 深度学习-06(PaddlePaddle体系结构与基本概念[Tensor、Layer、Program、Variable、Executor、Place]线性回归、波士顿房价预测)

    文章目录 深度学习-06(PaddlePaddle基础) paddlePaddle概述 PaddlePaddle简介 什么是PaddlePaddle 为什么学习PaddlePaddle PaddleP ...

  5. [Pytorch框架] 2.1.2 使用PyTorch计算梯度数值

    文章目录 使用PyTorch计算梯度数值 Autograd 简单的自动求导 复杂的自动求导 Autograd 过程解析 扩展Autograd import torch torch.__version_ ...

  6. Comparator之用最少数量的箭引爆气球

    文章目录 前言 关于Comparator 原题 前言 今天刷个题,遇到一个很有趣的问题,关于Comparator的使用,感觉也是一个关于写代码的一些小细节的问题 关于Comparator Compar ...

  7. Grafana 系列-统一展示-2-Prometheus 数据源

    系列文章 Grafana 系列文章 Grafana Prometheus 数据源 Grafana 提供了对 Prometheus 的内置支持.本文会介绍 Grafana Prometheus(也包括 ...

  8. google + chatgpt

    google注册 网址:https://www.google.com/ 使用右上角登录按钮 点击创建账户然后根据步骤注册 chagpt注册 1.https://chat.openai.com/auth ...

  9. 2021-08-11:按要求补齐数组。给定一个已排序的正整数数组 nums,和一个正整数 n 。从 [1, n] 区间内选取任意个数字补充到 nums 中,使得 [1, n] 区间内的任何数字都可以用

    2021-08-11:按要求补齐数组.给定一个已排序的正整数数组 nums,和一个正整数 n .从 [1, n] 区间内选取任意个数字补充到 nums 中,使得 [1, n] 区间内的任何数字都可以用 ...

  10. vue全家桶进阶之路15:自定义指令

    Vue 2.x 中的自定义指令是一种可以用于扩展 Vue.js 核心功能的特性.指令可以用于操作 DOM 元素的属性.监听 DOM 事件.控制 DOM 行为等等,可以将常见的交互行为封装成指令,从而让 ...