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

c++类型擦除

目录

一、核心原理

二、常见实现方式

1. 基于继承和虚函数(经典方式)

2. 基于 std::function(现代方式)

3. 基于模板和静态多态(编译时类型擦除)

三、应用场景

四、优缺点

优点:

缺点:

五、标准库中的类型擦除示例

总结


C++ 类型擦除(Type Erasure)是一种编程技术,用于在保持运行时多态的同时隐藏具体类型信息,使代码更灵活、更通用。它通过封装类型特定的实现细节,将动态类型转换的负担从编译时转移到运行时。以下从原理、实现方式和应用场景三个方面详细介绍:

一、核心原理

类型擦除的本质是分离接口与实现

  1. 定义统一接口:通过抽象基类或函数对象定义公共行为。
  2. 封装具体实现:将不同类型的实现细节封装在内部,对外提供统一接口。
  3. 运行时多态:通过虚函数或函数指针在运行时动态调用具体实现。

与传统继承多态的区别:

  • 继承多态:依赖公共基类,客户端需知晓具体派生类。
  • 类型擦除:客户端仅与接口交互,完全 unaware 具体类型。

二、常见实现方式

1. 基于继承和虚函数(经典方式)

通过抽象基类定义接口,派生类实现具体逻辑,外部使用智能指针管理对象。

示例代码

#include <iostream> // 添加对头文件的包含
#include <memory>
#include <string>// 定义接口
class Shape {
public:virtual ~Shape() = default;virtual double area() const = 0;virtual std::string name() const = 0;
};// 具体实现类
class Circle : public Shape {
public:explicit Circle(double r) : radius(r) {}double area() const override { return 3.14 * radius * radius; }std::string name() const override { return "Circle"; }
private:double radius;
};class Rectangle : public Shape {
public:Rectangle(double w, double h) : width(w), height(h) {}double area() const override { return width * height; }std::string name() const override { return "Rectangle"; }
private:double width, height;
};// 类型擦除包装器
class AnyShape {
public:template<typename T>AnyShape(T shape) : self(std::make_unique<Model<T>>(std::move(shape))) {}double area() const { return self->area(); }std::string name() const { return self->name(); }private:struct Concept {virtual ~Concept() = default;virtual double area() const = 0;virtual std::string name() const = 0;};template<typename T>struct Model : Concept {explicit Model(T value) : data(std::move(value)) {}double area() const override { return data.area(); }std::string name() const override { return data.name(); }T data;};std::unique_ptr<Concept> self;
};// 使用示例
int main() {AnyShape s1 = Circle(5.0);AnyShape s2 = Rectangle(3.0, 4.0);// 客户端无需知道具体类型,统一调用接口std::cout << s1.name() << ": " << s1.area() << std::endl; // 输出 "Circle: 78.5"std::cout << s2.name() << ": " << s2.area() << std::endl; // 输出 "Rectangle: 12"
}

关键点

  • AnyShape 是类型擦除包装器,内部持有指向 Concept 接口的指针。
  • Model<T> 是具体实现的适配器,将任意符合接口的类型 T 转换为统一的 Concept
2. 基于 std::function(现代方式)

利用 std::function 内置的类型擦除能力,直接存储可调用对象。

示例代码

#include <functional>
#include <string>
#include <vector>class AnyShape {
public:template<typename T>AnyShape(T shape) : area_func([shape]() { return shape.area(); }),name_func([shape]() { return shape.name(); }) {}double area() const { return area_func(); }std::string name() const { return name_func(); }private:std::function<double()> area_func;std::function<std::string()> name_func;
};// 使用示例
struct Triangle {double base, height;double area() const { return 0.5 * base * height; }std::string name() const { return "Triangle"; }
};int main() {std::vector<AnyShape> shapes;shapes.push_back(Triangle{3.0, 4.0});shapes.push_back(Circle{5.0}); // 复用前面的Circle类for (const auto& shape : shapes) {std::cout << shape.name() << ": " << shape.area() << std::endl;}
}

关键点

  • std::function 可以存储任何可调用对象(函数、lambda、成员函数等)。
  • 通过 lambda 捕获具体对象,将其方法转换为无参数的可调用对象。
3. 基于模板和静态多态(编译时类型擦除)

使用模板在编译时实现类型擦除,避免运行时开销。

示例代码

template<typename Shape>
double calculate_area(const Shape& shape) {return shape.area(); // 静态多态:依赖Shape类型有area()方法
}// 使用示例
struct Square {double side;double area() const { return side * side; }
};int main() {Square s{5.0};std::cout << calculate_area(s) << std::endl; // 编译时确定类型
}

特点

  • 编译时多态,无虚函数调用开销。
  • 要求所有类型实现统一接口(鸭子类型)。

三、应用场景

  1. 泛型容器:存储不同类型但具有相同接口的对象。

  • 例如:std::vector<AnyShape> 可存储 Circle、Rectangle 等任意形状。
  1. 插件系统:动态加载不同实现,统一接口调用。

  • 例如:游戏引擎加载不同厂商的渲染插件。
  1. 回调函数封装:隐藏回调函数的具体类型。

  • 例如:GUI 框架的事件处理系统,存储不同类型的事件回调。
  1. 跨库接口:在不同库之间传递对象,避免暴露具体类型。

  • 例如:数据库驱动返回统一的 RecordSet 接口,隐藏底层实现。

四、优缺点

优点
  • 解耦接口与实现:客户端无需依赖具体类型,降低编译依赖。
  • 灵活性:可在运行时动态切换实现。
  • 代码复用:同一套逻辑处理多种类型。
缺点
  • 性能开销:虚函数调用或 std::function 的间接调用会降低性能。
  • 类型信息丢失:无法恢复原始类型(除非手动存储类型标签)。
  • 实现复杂度:需要设计额外的包装层,代码可读性可能降低。

五、标准库中的类型擦除示例

  • std::function:可存储任何可调用对象,擦除具体函数类型。
  • std::any:可存储任何类型的值,运行时查询类型。
  • std::vector<bool>:特殊实现,擦除了 bool 的真实类型(使用位压缩)。

总结

类型擦除是 C++ 中实现 “运行时泛型” 的强大技术,通过隐藏具体类型信息,提供统一接口,使代码更灵活、更具扩展性。选择合适的实现方式(继承、std::function 或模板)取决于具体场景的性能需求和类型安全要求。

http://www.xdnf.cn/news/1054189.html

相关文章:

  • DNS递归查询步骤
  • 2. Anaconda 的安装及 Pytorch 环境安装
  • 2 Studying《Arm A715 Technical Reference Manual》
  • Java 常用类 Arrays:从零到实战的数组操作指南
  • 第五节:Vben Admin 最新 v5.0 (vben5) 快速入门 - 角色管理模块(上)
  • 云知声“流血”上市:三年亏损超12亿元,负债高企,现金流紧张
  • 进程间通信之进程间传递文件描述符
  • 【杂谈】-剖析 LLMs 与 LRMs:人工智能推理的困境与展望
  • 深度学习---ONNX(Open Neural Network Exchange)
  • python zip() 函数的用法
  • 《一元线性回归:从基础到应用及模型处理》
  • centos7安装weblogic
  • linux多线程之线程基础
  • ATSAMV71Q21B基于Microchip Studio以及ASF4.0架构使用printf打印float类型
  • 超标量处理器设计9-执行
  • 647. 回文子串
  • AI驱动SEO关键词精准布局
  • PMP成本管理时,合同成本的计算和注意事项
  • 耗时3小时,把这两天做好的爬虫程序,用Python封装成exe文件
  • 构建高性能日志系统:QGroundControl日志模块深度解析
  • 【JavaEE】(2) 多线程1
  • 第3章 C#编程概述 笔记
  • 计算机求职提前批/求职什么时候投递合适
  • 宝塔部署.net项目(nopcommerce)
  • K-Means算法详细解析:从原理到实践
  • C++ STL常用二分查找算法
  • 2025年品牌定位推荐排行榜:锚定市场航向,解锁品牌增长新势能
  • Python+QT远程控制助手-ver2
  • 《注解的江湖:一场元数据的“宫斗剧”》
  • 每日算法刷题Day32 6.15:leetcode枚举技巧7道题,用时1h10min