手写 C++ Any 类:深入理解多态与模板(示例详解)

 更新时间:2026年05月28日 10:39:25   作者:WWTYYDS_666  
这段代码详细介绍了使用C++实现类型擦除的类Placeholder,通过抽象基类和模板子类实现类型安全的存储与操作,强调了RAII机制和拷贝交换法在内存管理中的关键作用,感兴趣的朋友一起看看吧

我们自己实现的阉割版 Any 类核心能力:用一个类存储任意类型数据(int/自定义类/字符串等),底层依靠 抽象基类+模板子类 实现类型擦除

模块1:头文件包含

#include <iostream>   // 输入输出
#include <typeinfo>   // 类型识别:typeid 运算符核心依赖
#include <utility>    // 标准库 swap 函数
#include <cassert>    // 断言:类型不匹配时报错

讲解

这四个头文件是实现 Any 类的基础:

  • typeinfo:提供 typeid,用于类型安全校验
  • utility:提供 std::swap,实现高效交换;
  • cassert:断言,防止非法类型获取数据,保证程序安全。

模块2:内部抽象基类holder(类型擦除核心接口)

class holder
{
public:
    virtual ~holder(){}  // 虚析构函数
    virtual const std::type_info& type() =0; // 纯虚函数:获取类型
    virtual holder* clone()=0; // 纯虚函数:克隆对象
};

讲解

  1. 定位Any 类的私有内部抽象类,对外完全隐藏;
  2. 虚析构:必须加!保证多态删除子类对象时,析构函数能被正确调用,防止内存泄漏;
  3. 纯虚函数
    • type():返回存储数据的类型信息;
    • clone():深拷贝对象,支持 Any 的拷贝/赋值;
  4. 作用:定义统一接口,屏蔽具体类型差异,是类型擦除的基石。

模块3:模板子类placeholder<T>(真正存储数据的地方)

template<class T>
class placeholder:public holder
{
public:
    placeholder(const T&val):_val(val){}
    virtual const std::type_info& type(){return typeid(T);}
    virtual holder* clone(){return new placeholder(_val);}
public:
    T _val; // 真实存储数据的变量
};

讲解

  1. 模板类:T 是任意类型(int/Test/string…),真正实现存储任意数据;
  2. 继承 holder:满足多态要求,对外统一表现为 holder 类型;
  3. 构造函数:初始化成员变量 _val,存入数据;
  4. 重写纯虚函数:
    • type():返回当前模板类型 T 的信息;
    • clone():创建自身副本,实现深拷贝;
  5. 核心:_val 是真实数据存储位置,外部完全不可见。

模块4:Any类核心成员 + 构造/析构函数

holder*_content; // 核心指针
// 默认构造:空对象
Any():_content(nullptr){}
// 模板构造函数:接收任意类型数据
template<class T>
Any(const T&val):_content(new placeholder<T>(val)){}
// 拷贝构造函数:深拷贝
Any(const Any&other):_content(other._content?other._content->clone():nullptr)
{}
// 析构函数:自动释放内存
~Any(){delete _content;}

讲解

  1. 核心成员 _content
    holder 基类指针,多态指向 placeholder<T> 子类对象,实现类型擦除;
  2. 默认构造:创建空的 Any 对象;
  3. 模板构造:接收任意类型 T,创建 placeholder<T> 对象并赋值给指针;
  4. 拷贝构造:调用 clone() 实现深拷贝,避免浅拷贝指针悬空;
  5. 析构函数:RAII 机制,自动释放内存,无需手动管理指针。

模块5:交换函数swap(异常安全核心)

Any&swap(Any&other)
{
    std::swap(_content,other._content);
    return *this;
}

讲解

  1. 作用:避免自赋值,先构造一个临时对象,再交换交换两个 Any 对象的内部指针;
  2. 价值:是拷贝交换法的基础,让赋值运算符异常安全、无内存泄漏、无需处理自赋值
  3. 极简高效:仅交换指针,不拷贝数据,性能极高。

模块6:数据获取函数get<T>()(类型安全)

template<class T>
T*get()
{
    assert(typeid(T)==_content->type());
    return &((placeholder<T>*)_content)->_val;
}

讲解

  1. 模板函数:指定要获取的类型 T
  2. 断言校验:类型不匹配直接终止程序,保证类型安全
  3. 类型强转:将基类指针强转为子类指针,获取真实数据 _val
  4. 返回值:返回数据指针,可读写内部数据。

模块7:赋值运算符重载(两种赋值方式)

// 赋值1:任意类型直接赋值给 Any
template<class T>
Any& operator=(const T&val)
{
    Any(val).swap(*this);
    return *this;
}
// 赋值2:Any 对象之间赋值
Any& operator=(const Any&other)
{
    Any(other).swap(*this);
    return *this;
}

讲解

  1. 统一用法拷贝交换法,C++ 赋值运算符最优写法;
  2. 原理
    ① 先创建临时对象 Any(val) / Any(other)
    ② 交换临时对象和当前对象的指针;
    ③ 临时对象析构,自动释放旧内存;
  3. 优点:避免自赋值问题,异常安全, 无内存泄漏,代码极简

模块8:测试类Test(验证生命周期)

class Test
{
public:
    Test(){ std::cout<<"构造"<<std::endl;}
    Test(const Test&t){ std::cout<<"拷贝构造"<<std::endl;}
    ~Test(){ std::cout<<"析构"<<std::endl;}
};

讲解

自定义测试类,通过打印构造/拷贝构造/析构日志,验证:

  1. Any 类能正常存储自定义类型;
  2. 内存管理正确,无泄漏;
  3. 拷贝/赋值行为符合预期。

模块9:主函数测试(功能验证)

int main()
{
    Any a;
    {
        Test t;
        a=t;
    }
    a=10;
    int *pa=a.get<int>();
    std::cout<<*pa<<std::endl;
    return 0;
}

讲解

  1. 测试空对象初始化
  2. 测试自定义类型赋值
  3. 测试局部对象生命周期
  4. 测试int 类型赋值 + 获取数据
  5. 验证全程内存自动管理,无泄漏。

核心总结

  1. holder:抽象接口,定义统一规范;
  2. placeholder:模板子类,真正存数据;
  3. _content:基类指针,多态+类型擦除核心;
  4. 拷贝交换法:赋值运算符最优解;
  5. RAII:构造申请内存,析构自动释放。

到此这篇关于手写 C++ Any 类:深入理解多态与模板(示例详解)的文章就介绍到这了,更多相关 C++ Any 类内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 判断一个数是不是素数的方法

    判断一个数是不是素数的方法

    判断一个数是不是素数的方法,需要的朋友可以参考一下
    2013-03-03
  • C++自旋锁的实现示例

    C++自旋锁的实现示例

    自旋锁是一种非阻塞锁,线程在获取锁失败时会忙等待而不是挂起,它适用于锁持有时间极短的场景,可以在用户态运行,避免系统调用开销,下面就来详细的介绍一下C++自旋锁的使用,感兴趣的可以了解一下
    2026-02-02
  • C++创建多线程的方法总结

    C++创建多线程的方法总结

    下个迭代有个任务很有趣,用大量的线程去访问一个接口,直至其崩溃为止,这就需要多线程的知识,这也不是什么难事,本文总结一下C++中的多线程方法std、boost、pthread、windows api,感兴趣的朋友可以参考下
    2024-01-01
  • C++冒泡排序算法实例

    C++冒泡排序算法实例

    这篇文章主要介绍了C++冒泡排序算法实例,本文先是介绍了什么是冒泡排序,然后给出了实现代码,需要的朋友可以参考下
    2014-10-10
  • 详解C++中的万能头文件

    详解C++中的万能头文件

    C++万能头文件它是一个包含了每一个标准库的头文件,接下来通过本文给大家介绍C++中的万能头文件及优缺点,需要的朋友可以参考下
    2023-02-02
  • 详解Qt中的双缓冲机制与实例应用

    详解Qt中的双缓冲机制与实例应用

    所谓双缓冲机制,是指在绘制控件时,首先将要绘制的内容绘制在一个图片中,再将图片一次性地绘制到控件上。本文主要为大家介绍了Qt中的双缓冲机制与实例应用,希望对大家有所帮助
    2023-03-03
  • 八皇后问题的相关C++代码解答示例

    八皇后问题的相关C++代码解答示例

    这篇文章主要介绍了八皇后问题的相关C++代码解答示例,文中包括ACM竞赛的八皇后相关知识的练习实例,需要的朋友可以参考下
    2015-08-08
  • C语言实现宾果消消乐

    C语言实现宾果消消乐

    这篇文章主要为大家详细介绍了C语言实现宾果消消乐,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-12-12
  • C++使用new操作符动态分配的实现

    C++使用new操作符动态分配的实现

    本文主要介绍了C++使用new操作符动态分配的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2025-06-06
  • 浅谈c++中的stl中的map用法详解

    浅谈c++中的stl中的map用法详解

    下面小编就为大家带来一篇浅谈c++中的stl中的map用法详解。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-10-10

最新评论