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

条款25:考虑写出一个不抛出异常的swap函数

1.前言

swap是个有趣的函数,原本它只是STL的一部分,而后成为异常安全编程的基石,以及用来处理自我赋值可能性的一个常见机制。由于swap功能如此强大,适当的实现很重要。然而在非凡的重要性之外它也带来了非凡的复杂度。

所谓swap两对象值,意思是将两对象的值彼此赋予对方。缺省情况下swap动作可由标准程序库提供的swap算法完成。其典型的实现方式如下:

namespace std{template<typename T>void swap(T& A,T& b){T temp(a);a=b;b=temp;}
}

只要类型T支持copying(通过copy构造函数和copy assignment操作符完成),缺省的swap实现代码就会帮你置换类型为T的对象,不需要为此做另外的工作。

该缺省的swap实现版本十分平淡,无法刺激你的肾上腺。它涉及到三个对象的复制:a复制到temp,b复制到a,以及temp复制到b。但是对于某些类型而言,这些复制动作没有必要:对那种情况而言swap缺省行为等于杀机用牛刀。

2.实例

最主要的一种类型就是“以指针指向一个对象,内含真正数据”那种类型。这种设计的常见表现形式就是所谓的“pimpl手法”。如果以这种手法设计Widget class,看起来像这样:

class WidgetImpl{   //针对Widget数据而设计的classpublic:
...
private:
int a,b,c;//可能有许多数据
std::vector<double> v;//意味复制时间很长
。。。};
class Widget{public:Widget(const Widget& rhs);Widget& operator=(const Widget& rhs){...*pImpl=*(rhs.pImpl);}...private:Widget Impl* pImpl;//指针,所有对象内含Widget
}

一旦要置换两个Widget对象值,我们唯一需要做的就是置换其pImpl指针,但缺省的swap算法不知道这一点,它不止复制三个widgets,还复制三个WidgetImpl对象。非常缺乏效率。

我们希望能够告诉std::swap,当widget被置换时,真正该做的是置换其内部的pImpl指针。确切实践这个思路的一个做法是:将std::swap针对Widget特化,下面是基本的构想,但目前这个形式无法通过编译:

namespace std{template<>void swap<Widget>(Widget& a,Widget& b){swap(a.pImpl,b.pImpl);}
}    

这个函数一开始得"template<>"表示它是std::swap的一个全特化版本,函数名称之后的"<Widget>"表示这一特化版本系针对“T是Widget”而设计。换句话说当一般性的swap template施行于Widget身上便会启用这个版本。通常我们不能够改变std命名空间内的任何东西,但可以为标准template制造特化版本,使它专属于我们自己的classes。

但是该函数无法通过编译,因为它企图访问a和b内的pImpl指针,而那却是private,我们可以将这个特化版本声明为friend,但和以往的规矩不太一样:我们令Widget声明为swap的public成员函数做真正的置换工作,然后将std::swap特化,令它调用该成员函数:

class Widget{public:...void swap(Widget& other){using std::swap;swap(pImpl,other.pImpl);//若要置换Widgets就置换其pImpl指针}...
};
namespace std{template<>void swap<Widget>(Widget& a,Widget& b){a.swap(b);//若要置换Widgets,调用其swap成员函数}
}

这种做法不只能够通过编译,还与STL容器有一致性,因为所有STL容器也都提供有public swap成员函数和std::swap特化版本。

然而假设Widget和WidgetImpl都是class templates而非classes,也许我们可以将Widget内的数据类型加以参数化:

template<typename T>
class WidgetImpl{...};
template<typename T>
class Widget{....};

http://www.lryc.cn/news/258189.html

相关文章:

  • linux 中crontab 定时任务计划创建时间文件夹示例
  • 欣赏动态之美,不如欣赏C语言实现动态内存管理之美 ! ! !
  • from pycocotools.coco import COCO报错
  • CentOS服务自启权威指南:手动启动变为开机自启动(以Jenkins服务为例)
  • 第二百零一回 介绍一个三方包open_settings
  • iview Table实现跨页勾选记忆功能以及利用ES6的Map数据结构实现根据id进行对象数组的去重
  • 【Spring 源码】 贯穿 Bean 生命周期的核心类之 AbstractAutowireCapableBeanFactory
  • 漏洞复现-某友UFIDA NC系统某接口未授权访问漏洞(附漏洞检测脚本)
  • 树莓派5安装opencv
  • 【测试开发】Python+Django实现接口测试工具
  • 从 MQTT、InfluxDB 将数据无缝接入 TDengine,接入功能与 Logstash 类似
  • 友元c++
  • java: 错误: 不支持发行版本 6
  • qml刷新C++中的QImage图像
  • IJCAI 2024 International Joint Conference on Artificial Intelligence
  • 使用Python Flask搭建Web问答应用程序并发布到公网远程访问
  • android 13.0 app应用安装白名单
  • SSL证书HTTPS保护服务
  • 快速认识什么是:Docker
  • c语言青蛙跳台阶
  • IntelliJ IDEA 2023.3 最新版如何试用?IntelliJ IDEA 2023.3 最新版试用方法
  • Java参数校验详解:使用@Valid注解和自定义注解进行参数验证
  • 多维时序 | MATLAB实现BWO-CNN-BiGRU-Multihead-Attention多头注意力机制多变量时间序列预测
  • C++ 中的引用
  • MQ-Det: Multi-modal Queried Object Detection in the Wild
  • HarmonyOS应用开发初体验
  • 《C++新经典设计模式》之第4章 策略模式
  • 【方法】PowerPoint“只读方式”如何取消?
  • MySQL数据库概念与实践
  • 【ArcGIS Pro微课1000例】0052:基于SQL Server创建企业级地理数据库案例