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

[Effective C++]条款45 运用成员函数模板接受所有兼容类型

本文初发于 “天目中云的小站”,同步转载于此。

条款45 : 运用成员函数模板接受所有兼容类型

本条款中我们将会以智能指针为例, 介绍如何通过成员函数模板使一个模板类可以接受所有兼容类型.

我们先来构建一个简单的继承体系 :

class Top { ... };
class Middle: public Top { ... };
class Bottom: public Middle { ... };
Top *pt1 = new Middle;                   // Middle*隐式转换为Top*
Top *pt2 = new Bottom;                   // Bottom*隐式转换为Top*
const Top *pct2 = pt1; 					 // Top*隐式转换为const Top*

在本例中, Top是最初始的基类, 依次派生出MiddleBottom, 通过隐式转换, 各种指针类型是可以合理地进行隐式转换的. 但当我们想用智能指针代行管理事务时, 再想这样的转换就比较麻烦了, 当然标准库中的智能指针已经解决了这种问题, 我们现在要讨论的就是标准库是如何实现智能指针之间的隐式转换的, 我们期待的效果如下 :

template<typename T>
class SmartPtr {
public:                             
  explicit SmartPtr(T *realPtr);    // 通过资源指针进行初始化
  ...
};

SmartPtr<Top> pt1 = SmartPtr<Middle>(new Middle);   
SmartPtr<Top> pt2 = SmartPtr<Bottom>(new Bottom); 
SmartPtr<const Top> pct2 = pt1; 

这段代码是无法通过编译的, 因为就算是TopMiddle有联系, SmartPtr<Top>SmartPtr<Middle>也没有任何联系, 它们是无法隐式转换的, 但是我们可以通过成员函数模板, 具体说是写一个泛化copy构造函数来创造这种联系.


泛化copy构造函数

我们先来写一个成员函数模板中的泛化copy构造函数 :

template<typename T>
class SmartPtr {
public:
  template<typename U>                       // 泛化copy构造函数
  SmartPtr(const SmartPtr<U>& other);       
  ...                 
};

这个模板函数接受用一个SmartPtr<U>类型的参数去构造一个SmartPtr<T>类型的对象, 现在还只是声明, 我们应该考虑如何定义内部逻辑, 正常逻辑应该是先看U*是否可以隐式转换为T*, 如果可以转换也就可以进行智能指针之间的转换, 我们来看代码 :

template<typename T>
class SmartPtr {
public:
  template<typename U>
  SmartPtr(const SmartPtr<U>& other)         
  : heldPtr(other.get()) {...}            // 关键代码
    
  T* get() const { return heldPtr; }
  ...
private:                                     
  T *heldPtr;                                
};

这里直接将参数other中的heldPtr取出, 赋值给当前对象的heldPtr. 其实在这里就进行了检查 :

  • other.get()取出的指针类型为U*, 如果U*可以隐式转换为T*, 那么可以进行智能指针之间的转换.
  • 如果不可以隐式转换, 编译错误, 会被系统拦截.

至此, 通过泛化copy构造函数这个成员函数模板, 只要U*可以隐式转换为T*, 那么SmartPtr<U>也可以隐式转换为SmartPtr<T>.


成员函数模板

成员函数模板的效用不只局限于构造函数, 也可以支持赋值操作, 其不改变语言规则, 但是可以帮助你让class在构造和赋值操作上可以兼容更多类型, 让我们在使用模板类型时可以像使用非模板类型时一样自然流畅. 我们可以了解一下标准库中shared_ptr的简略版本, 看看其对成员函数模板的使用 :

template<class T> 
class shared_ptr {
public:
  // 构造
  template<class Y>                                     
    explicit shared_ptr(Y * p);                         // 泛化构造函数
  template<class Y>                                     
    shared_ptr(shared_ptr<Y> const& r);                 // 泛化copy构造函数
  template<class Y>                                     
    explicit shared_ptr(weak_ptr<Y> const& r);          // 通过weak_ptr构造
  template<class Y>
    explicit shared_ptr(unique_ptr<Y>& r);				// 通过unique_ptr构造
  // 赋值
  template<class Y>                                    
    shared_ptr& operator=(shared_ptr<Y> const& r);      // 泛化赋值重载
  template<class Y>                                     
    shared_ptr& operator=(unique_ptr<Y>& r);            // 用unique_ptr赋值
  ...
};

在本例中, 构造函数中只有泛化copy构造函数没有explicit, 说明其他构造函数都不应当支持隐式类型转换, 也就是说T和Y的类型应当一致. 另外成员模板函数并不改变语言规则, 就算我们写了泛化的拷贝构造函数和赋值重载, 依旧不影响普通的拷贝构造和赋值重载, 如果我们没有写, 编译器还是会自动生成, 所以如果想要控制构造的方方面面, 我们应当同时声明普通版本和泛化版本.

template<class T> 
class shared_ptr {
public:
  shared_ptr(shared_ptr const& r);                 // 普通拷贝构造
  template<class Y>                                
    shared_ptr(shared_ptr<Y> const& r);            // 泛化拷贝构造

  shared_ptr& operator=(shared_ptr const& r);      // 普通赋值重载
  template<class Y>                               
    shared_ptr& operator=(shared_ptr<Y> const& r); // 泛化赋值重载
  ...
};

请记住 :

  • 使用成员函数模板可以生成"可接受所有兼容类型"的函数.
  • 当我们声明泛化拷贝构造和赋值重载时, 也应该声明其普通版本.

http://www.kler.cn/a/502145.html

相关文章:

  • B+树的原理及实现
  • 哈夫曼、算术、LZ编码
  • 【Linux网络编程】数据链路层 | MAC帧 | ARP协议
  • 深度优先和广度优先【栈、堆前端举例】
  • Idea-离线安装SonarLint插件地址
  • 后端开发 Springboot整合Redis Spring Data Redis 模板
  • 如何使用 Java 的 Spring Boot 创建一个 RESTful API?
  • c++ 中的容器 vector、deque 和 list 的区别
  • 穿越火线怀旧服预约网页vue3版本
  • JavaScript 类型转换
  • EFK采集k8s日志
  • 【OpenGL/C++】面向对象扩展——测试环境
  • FlashAttention的原理及其优势
  • HTTP/HTTPS ④-对称加密 || 非对称加密
  • 使用WeakHashMap实现缓存自动清理
  • 特制一个自己的UI库,只用CSS、图标、emoji图 第二版
  • MySQL Binlog 同步工具go-mysql-transfer Lua模块使用说明
  • Django创建数据表、模型、ORM操作
  • 饿汉式单例与懒汉式单例模式
  • 前端学习-事件对象与典型案例(二十六)
  • 25/1/13 算法笔记<嵌入式> 继续学习Esp32
  • uiautomator2 实现找图点击
  • 记一次学习skynet中的C/Lua接口编程解析protobuf过程
  • FreeSWITCH Sofia SIP 模块常用命令整理
  • 如何设计一个 RPC 框架?需要考虑哪些点?
  • 计算机网络 笔记 网络层1