详解 C++ 中移动构造与移动赋值

1. 基本介绍

  • 移动赋值语法原理
    移动赋值语法的原理是基于右值引用(rvalue reference)的概念。在C++11中引入了右值引用,通过使用双 ampersands(&&)来声明一个右值引用。

  • 右值引用
    右值引用可以绑定到临时对象(即右值),例如字面量、临时对象或者表达式的结果。与左值引用不同,右值引用可以被修改,因此可以将资源从一个对象转移到另一个对象。

  • 移动赋值运算符
    移动赋值运算符 (operator=)是一种特殊的成员函数,允许将一个对象的资源从另一个对象“移动”到当前对象而不是进行拷贝。它通常被实现为接受一个参数为右值引用类型的函数。

  • 移动赋值的实现步骤
    在实现移动赋值运算符时,通常需要执行以下步骤:
    a. 检查自我赋值:使用 if (this != &other) 确保不是自我赋值。
    b. 通过使用 std::move() 将资源从其他对象移动到当前对象。
    c. 将其他对象中该资源的状态置为适当的默认状态。例如,在基本类型中将其设为0,在类中可能需要特定的操作。
    d. 返回当前对象的引用。

  • 移动赋值优势
    移动赋值语法使得在某些情况下能够更高效地管理资源而无需进行额外的数据拷贝。这对于大型数据结构或具有显式资源管理的类特别有用,例如动态分配的内存或文件句柄等。

2. 代码示例:

下面是一个使用移动赋值的类及其子类的示例:
构建一个基类,一个子类,基类包含移动构造与移动赋值,同时子类中也包含移动构造与移动赋值操作;

#include <iostream>class Base {
public:Base(int data) : mData(data) {}// 移动构造函数Base(Base&& other) noexcept {std::cout << "Base Move constructor" << std::endl;mData = std::move(other.mData);other.mData = 0;}// 移动赋值运算符Base& operator=(Base&& other) noexcept {std::cout << "Base Move assignment operator" << std::endl;if (this != &other) {mData = std::move(other.mData);other.mData = 0;}return *this;}protected:int mData;
};class Derived : public Base {
public:Derived(int data1, int data2, int extraData): Base(data1), mMoreData(data2), mExtraData(extraData) {}// 移动构造函数Derived(Derived&& other) noexcept : Base(std::move(other)) {std::cout << "Derived Move constructor" << std::endl;mMoreData = std::move(other.mMoreData);other.mMoreData = 0;mExtraData = std::move(other.mExtraData);other.mExtraData = 0;}// 移动赋值运算符Derived& operator=(Derived&& other) noexcept {std::cout << "Derived Move assignment operator" << std::endl;if (this != &other) {Base::operator=(std::move(other));mMoreData = std::move(other.mMoreData);other.mMoreData = 0;mExtraData = std::move(other.mExtraData);other.mExtraData = 0;}return *this;}private:int mMoreData;int mExtraData;
};int main() {// 创建一个派生类对象并赋值给另一个派生类对象Derived derived1(20, 30, 40);Derived derived2 = std::move(derived1);return 0;
}

3. 运行结果:

运行这段代码,输出结果为:

Base Move constructor
Derived Move constructor
在这个示例中,Derived 类添加了一个成员变量 mExtraData。在移动构造函数和移动赋值运算符中,我们需要先调用基类的对应函数来处理基类的成员变量,然后再处理派生类的成员变量。在移动构造函数和移动赋值运算符中,必须确保不会忘记调用基类的相应函数。

4. 特别注意事项:

请注意,在移动赋值运算符中,我们需要检查自我赋值,并在释放资源之前将原对象的成员变量设置为适当的值。此外,在派生类的移动构造函数和移动赋值运算符中也需要处理派生类的额外成员变量。

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

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

相关文章

从 React 到 Qwik:开启高效前端开发的新篇章

1. Qwik Qwik 是一个为构建高性能的 Web 应用程序而设计的前端 JavaScript 框架,它专注于提供即时启动性能,即使是在移动设备上。Qwik 的关键特性是它采用了称为“恢复性”的技术,该技术消除了传统前端框架中常见的 hydration 过程。 恢复性是一种序列化和恢复应用程序状态…

日常学习之:如何使用 dockerfile 将 vue 的单独前端项目通过 docker 的方式部署到 heroku上

文章目录 需求描述开始操作准备阶段&#xff1a;准备 server.js 文件并安装依赖&#xff0c;将 vue 项目包装成单独的服务器制作 server.js安装 server.js 需要的依赖 构建 Dockerfileheroku container 链接和部署其他细节 需求描述 你想用 vue 构建前端&#xff0c;用 django…

设计模式六(模板方法模式)

是一种行为型的设计模式&#xff0c;将一些步骤的实现延迟到子类有些步骤是固定的可以抽取父类&#xff0c;在父类中实现各个步骤&#xff0c;并且在父类中定义一个流程控制方法&#xff0c;将某个容易发生变化的步骤定义为抽象方法&#xff0c;针对不同的子类实现不同的方法&a…

【设计模式】阿里终面:你觉得这个例子是策略模式吗?

什么是策略模式&#xff1f; 策略模式&#xff0c;举几个贴近生活的例子&#xff1a;当我们出行的时候&#xff0c;不同的出行方式就是不同的策略&#xff0c;例如走路、开车、骑自行车、坐飞机、坐邮轮等等&#xff0c;每一种出行方式都代表着不同的费用和时间&#xff1b;当…

关于css 的基础试题

CSS是什么的缩写&#xff1f; A. Creative Style SheetsB. Cascading Style SheetsC. Computer Style SheetsD. Colorful Style Sheets 在HTML中&#xff0c;通过什么标签引入CSS样式&#xff1f; A. <script>B. <style>C. <link>D. <css> 以下哪个选项…

make: *** No rule to make target ‘clean‘. Stop.

项目场景&#xff1a; 在Ubuntu下编写makefile文件编译的时候,出现make: *** No rule to make target ‘clean’. Stop. 问题描述 make: *** No rule to make target ‘clean’. Stop. 解决方案&#xff1a; 原本我makefile文件的名字是MakeFile , 把它改为makefile以后完美运…

腾讯云OpenCloudOS安装ES(elasticsearch7.17.16)

腾讯云OpenCloudOS安装ES&#xff08;elasticsearch7.17.16&#xff09; 下载ES 先从官网下载es的Linux解压包官网地址 https://www.elastic.co/cn/downloads/past-releases/elasticsearch-7-17-16 下载完成后&#xff0c;将其放置在自己想要放到的路径下 配置ES 解压文件 …

第五季特别篇:一夜杯、游戏之宴 2017.04.26

第五季特别篇&#xff1a;一夜杯、游戏之宴 2017.04.26 OVA 第1话&#xff1a;一夜酒杯 / 一夜杯OVA 第2话&#xff1a;游戏之宴 / 遊戯の宴 OVA 第1话&#xff1a;一夜酒杯 / 一夜杯 遭到独角妖袭击的妖怪夫妇日土和初菜被夏目所救&#xff0c;这对妖怪夫妇制作的酒杯&#xf…

R数据分析:非劣效性研究设计的统计处理方法,原理和实例

在我们经常接触的统计模式中&#xff0c;我们是在寻求推翻原假设&#xff0c;证明差异&#xff0c;这种统计模型在传统的临床试验中&#xff0c;在各种统计推断中已经成为默认了。在传统的临床试验中通常会将一种新的治疗方法与标准治疗或安慰剂进行比较&#xff0c;从而证明这…

【Spring Boot 3】【@Scheduled】动态删除定时任务

【Spring Boot 3】【@Scheduled】动态删除定时任务 背景介绍开发环境开发步骤及源码工程目录结构总结背景 软件开发是一门实践性科学,对大多数人来说,学习一种新技术不是一开始就去深究其原理,而是先从做出一个可工作的DEMO入手。但在我个人学习和工作经历中,每次学习新技…

###C语言程序设计-----C语言学习(5)#

前言&#xff1a;感谢您的关注哦&#xff0c;我会持续更新编程相关知识&#xff0c;愿您在这里有所收获。如果有任何问题&#xff0c;欢迎沟通交流&#xff01;期待与您在学习编程的道路上共同进步&#xff01; 一. 主干知识的学习 1.switch语句 switch语句可以处理多分支选…

Linux详细笔记大全

第0章 Linux基础入门 什么是计算机 计算机的组成: 控制器,是整个计算机的中枢神经,根据程序要求进行控制,协调计算机各部分工作及内存与外设的访问等。 运算器,功能是对数据进行各种算术运算和逻辑运算。 存储器,功能是存储程序、数据和各种信号、命令等信息。 输入设备…

LeetCode链表总结

206.反转链表 两种方法&#xff0c;一种是迭代法&#xff08;头插法&#xff09;&#xff0c;一种是递归法。头插法比较简单&#xff0c;就不再多说了&#xff0c;讲下递归法。主要是要注意递归以后head->next指向的是哪个结点&#xff0c;其实head->next的指向是不变的…

第二模块 函数模块

第二模块 函数&模块 day09 文件操作相关1. 文件操作1.1 读文件1.2 写文件1.3 文件打开模式1.4 常见功能1.5 上下文管理练习题 2.csv格式文件3.ini格式文件4.XML格式文件4.1 读取文件和内容4.2 读取节点数据4.3 修改和删除节点4.4 构建文档 5.Excel格式文件5.1 读Excel5.1 写…

微信小程序之页面导航、生命周期和WXS脚本

学习的最大理由是想摆脱平庸&#xff0c;早一天就多一份人生的精彩&#xff1b;迟一天就多一天平庸的困扰。各位小伙伴&#xff0c;如果您&#xff1a; 想系统/深入学习某技术知识点… 一个人摸索学习很难坚持&#xff0c;想组团高效学习… 想写博客但无从下手&#xff0c;急需…

STM32F407移植OpenHarmony笔记2

接上篇&#xff0c;搭建完开发环境后&#xff0c;我们还要继续工作。 官方合作的开发板刚好有STM32F407&#xff0c;我准备试一下开发板的demo&#xff0c;虽然我用的不是他们的开发板。 先下载以下3份代码&#xff1a; https://gitee.com/openharmony/device_board_talkweb…

Python 使用重构重命名一键更改变量名的方法

一个变量有多处引用的情况下&#xff0c;需要重命名&#xff0c;可以使用重构重命名进行一键更改。 方法是:选择变量名–>右键–>Refactor–>Rename&#xff08;也可以使用快捷&#xff1a;选择变量后按下ShiftF6&#xff09;&#xff0c;然后直接输入新的变量名即可…

基于Matlab/Simulink直驱式风电储能制氢仿真模型

接着还是以直驱式风电为DG中的研究对象&#xff0c;上篇博客考虑的风电并网惯性的问题&#xff0c;这边博客主要讨论功率消纳的问题。 考虑到风速是随机变化的&#xff0c;导致风电输出功率的波动性和间歇性问题突出&#xff1b;随着其应用规模的不断扩大以及风电在电网中渗透率…

[echarts] 图表工具栏 toolbox

option{// 工具栏配置toolbox:{id:1, // 组件IDshow:true, // 是否显示工具栏orient:horizontal, // 工具栏 icon 的布局朝向itemSize:15, // 工具栏 icon 的大小itemGap:10, // 工具栏…

DjangoURL调度器(二)

一、默认值与额外参数 1.1、默认值 1.1.1、urls.py from django.urls import pathfrom . import viewsurlpatterns [# http://127.0.0.1:8000/polls/blog/ 等同于 # http://127.0.0.1:8000/polls/blog/1/path(blog/, views.page),# http://127.0.0.1:8000/polls/blo…