当前位置: 首页 > news >正文

C++ 拷贝交换技术示例

拷贝交换技术(copy and swap)是什么,网上估计能查到很多。但网上有点难找到完整的演示代码,所以这里记录一下。难点在于:

  1. 如果要满足 5 的原则,我到底要写那些函数?
    默认构造函数、复制构造函数、析构函数、swap 函数。剩下三个函数是固定模板(boilerplate),不用写与类相关的代码。由于两种重载赋值运算符合并成一个了,所以只剩两个函数需要写固定模板。
  2. 哪些是 noexcept 的?
    必须是 noexcept 的函数:移动构造函数,重载赋值运算符(注意只剩一个了),交换函数。
  3. 交换函数和 std::swap 的关系如何?
    必须自己写一个属于这个类的交换函数,在实现拷贝交换技术时不能调用 std::swap。在类外面,可以通过 using std::swap; 的方法,让编译器优先选用自己实现 swap 函数,如不存在,再回退到 std::swap;在类外面也可以直接都用 std::swap,无所谓。
#include <cassert> // assert
#include <utility> // swapclass Demo {using Self = Demo;public:int ichi{}; // Note 1: 这里初始化为 0,又在默认构造函数初始化为 1,是为了说明委托默认构造函数的作用。特别地,如果存在成员没有用大括号初始化,就更需要委托默认构造函数完成所有成员的初始化。// 要么为 nullptr,要么为大小为 1 的数组。int* ein{};public:// Note 2: 默认构造函数是否是 constexpr, noexcept 的,视实际情况而定。constexpr Demo() noexcept : ichi{ 1 }, ein{ nullptr } {}// Note 3: 复制构造函数通常涉及内存分配,一般不是 noexcept 的。Demo(const Self& other) : Demo() { // Note 4: 需要委托默认构造函数,见 Note 1。不委托无法通过 assert。assert(ichi == 1);assert(ein == nullptr);ichi = other.ichi;if (other.ein) {ein = new int[ichi] { other.ein[0] };}}// Note 5: 使用拷贝交换技术,需要实现 swap 的函数。// Note 6: swap 函数应当是 noexcept 的。friend void swap(Self& a, Self& b) noexcept {// Note 7: 交换成员对象时,可以优先使用类型自己定义的友元函数 swap,不存在才回退到 std::swap。using std::swap;swap(a.ichi, b.ichi);swap(a.ein, b.ein);}// Note 8: 移动构造函数应当是 noexcept 的。Demo(Self&& other) noexcept : Demo() { // Note 9: 需要委托默认构造函数,见 Note 1。不委托无法通过 assert。assert(ichi = 1);assert(ein == nullptr);swap(*this, other); // Note 10: 使用拷贝交换技术时,必须使用自己定义的友元函数,绝不要使用 std::swap。}// Note 11: 重载赋值运算符可以只写一个,且应当是 noexcept 的。异常在复制构造函数中发生,不在赋值运算符中发生。Self& operator=(Self other) noexcept {swap(*this, other); // Note 12: 使用拷贝交换技术时,必须使用自己定义的友元函数,绝不要使用 std::swap。return *this;}// Note 13: 析构函数是否是 constexpr 的,视情况而定。constexpr ~Demo() { // Note 14: 析构函数默认总是 noexcept 的,不写。if (ein) {delete[] ein;}}
};int main() {// 测试 constexpr 默认构造。constexpr Demo default_obj;static_assert(default_obj.ichi == 1);// 测试默认构造。Demo allocated_obj;allocated_obj.ein = new int[1] { 114514 };{// 测试复制构造。Demo test_copy_constructor = allocated_obj;assert(test_copy_constructor.ein[0] == 114514);assert(allocated_obj.ein[0] == 114514);// 测试移动构造。Demo test_move_constructor = std::move(test_copy_constructor);assert(test_move_constructor.ein[0] == 114514);assert(test_copy_constructor.ein == nullptr);}{// 测试复制赋值。Demo test_copy_assignment;test_copy_assignment = allocated_obj;assert(allocated_obj.ein[0] == 114514);// 测试移动赋值。Demo test_move_assignment;test_move_assignment = std::move(test_copy_assignment);assert(test_move_assignment.ein[0] == 114514);assert(test_copy_assignment.ein == nullptr);}
}
http://www.lryc.cn/news/106551.html

相关文章:

  • 使用 Go 语言实现二叉搜索树
  • 系统接口自动化测试方案
  • 小研究 - JVM 垃圾回收方式性能研究(一)
  • [LeetCode]链表相关题目(c语言实现)
  • [深入理解NAND Flash (操作篇)] NAND 初始化常用命令:复位 (Reset) 和 Read ID 和 Read UID 操作和代码实现
  • RxJava 复刻简版之二,调用流程分析之案例实现
  • SpringMVC中Model和ModelAndView的区别
  • Tomcat安装与管理
  • React之路由
  • 机器学习深度学习——非NVIDIA显卡怎么做深度学习(坑点排查)
  • 2021 Robocom 决赛 第四题
  • 线程池-手写线程池Linux C简单版本(生产者-消费者模型)
  • 05-向量的意义_n维欧式空间
  • 交通运输安全大数据分析解决方案
  • vimrc 配置 (持续跟新中)
  • 【集成学习介绍】
  • 动画制作选择Blender还是Maya
  • 215. 数组中的第K个最大元素
  • NLP From Scratch: 生成名称与字符级RNN
  • Spring MVC程序开发
  • 医疗知识图谱问答——文本分类解析
  • JS关于多张图片上传显示报错不影响后面图片上传方法
  • MySQL踩坑之sql_mode的用法
  • 消息队列总结(4)- RabbitMQ Kafka RocketMQ高性能方案
  • websocket服务端大报文发送连接自动断开分析
  • 想写几个上位机,是选择学c#还是 c++ qt呢?
  • JavaScript 简单实现观察者模式和发布-订阅模式
  • java集成短信服务 测试版 qq邮箱简单思路
  • #P0994. [NOIP2004普及组] 花生采摘
  • Elasticsearch和Kibana的安装及验证