新闻中心
c++如何实现类型擦除(Type Erasure)_c++ std::any与std::function原理解析
类分离接口与具体类型,实现统一接口操作不同类型的机制,如std::any通过base_holder定义类型无关接口,holder模板保存具体值,并在运行时通过多态调用正确方法,从而实现任意类型存储与访问。
类型擦除是 C++ 中一种重要的编程技术,它允许你隐藏具体类型,从而实现更灵活的接口设计。典型的例子包括 std::any、std::function 和 std::variant。它们都使用了类型擦除来封装不同类型的数据或行为,对外提供统一的接口。
什么是类型擦除?
类型擦除的核心思想是:在编译期不知道具体类型的情况下,仍然能够存储和操作这些类型。它通过将“实际类型”与“接口”分离来实现。通常做法是:
- 定义一个统一的接口(如虚函数或多态基类)
- 用模板生成针对每种类型的实现
- 在运行时通过指针或引用调用正确的实现
这样使用者无需知道底层类型,就能完成操作。
手动实现一个简单的 std::any 风格类型擦除
std::any 可以保存任意类型的值。我们可以通过基类 + 模板派生类的方式来模拟其实现机制。
#include <memory>
#include <typeinfo>
#include <stdexcept>
class any {
private:
struct base_holder {
virtual ~base_holder() = default;
virtual const std::type_info& type() const = 0;
virtual std::unique_ptr<base_holder> clone() const = 0;
};
template<typename T>
struct holder : base_holder {
T value;
holder(const T& v) : value(v) {}
holder(T&& v) : value(std::move(v)) {}
const std::type_info& type() const override {
return typeid(T);
}
std::unique_ptr<base_holder> clone() const override {
return std::make_unique<holder>(value);
}
};
std::unique_ptr<base_holder> content;
public:
any() = default;
template<typename T>
any(const T& value) : content(std::make_unique<holder<T>>(value)) {}
any(const any& other)
: content(other.content ? other.content->clone() : nullptr) {}
any& operator=(const any& other) {
if (this != &other) {
content = other.content ? other.content->clone() : nullptr;
}
return *this;
}
any(any&&) = default;
any& operator=(any&&) = default;
bool has_value() const { return content != nullptr; }
const std::type_info& type() const {
return content ? content->type() : typeid(void);
}
template<typename T>
T& get() {
if (!content || content->type() != typeid(T)) {
throw std::bad_cast();
}
return static_cast<holder<T>&>(*content).value;
}
template<typename T>
const T& get() const {
if (!content || content->type() != typeid(T)) {
throw std::bad_cast();
}
return static_cast<const holder<T>&>(*content).value;
}
};
这个简化版的 any 使用多态基类 base_holder 来抹去具体类型。每个类型 T 实例化一个 holder<t></t>,保存真实值并重写虚函数。拷贝时通过 clone() 实现深拷贝。
std::function 的类型擦除原理
std::function<ret></ret> 能包装任何可调用对象(函数指针、lambda、bind 表达式等)。它的实现也依赖类型擦除。
核心思路与上面类似,但关注的是“调用”操作。我们需要:
Health AI健康云开放平台
专注于健康医疗垂直领域的AI技术开放平台
113
查看详情
- 一个通用调用接口
- 为每种可调用类型生成具体的执行逻辑
#include <memory>
#include <utility>
template<typename Signature>
class function;
template<typename Ret, typename... Args>
class function<Ret(Args...)> {
private:
struct callable_base {
virtual ~callable_base() = default;
virtual Ret call(Args... args) = 0;
virtual std::unique_ptr<callable_base> clone() const = 0;
};
template<typename F>
struct callable_wrapper : callable_base {
F func;
callable_wrapper(F f) : func(std::move(f)) {}
Ret call(Args... args) override {
return func(std::forward<Args>(args)...);
}
std::unique_ptr<callable_base> clone() const override {
return std::make_unique<callable_wrapper>(func);
}
};
std::unique_ptr<callable_base> impl;
public:
function() = default;
function(const function& other)
: impl(other.impl ? other.impl->clone() : nullptr) {}
function& operator=(const function& other) {
if (this != &other) {
impl = other.impl ? other.impl->clone() : nullptr;
}
return *this;
}
function(function&&) = default;
function& operator=(function&&) = default;
template<typename F>
function(F f) : impl(std::make_unique<callable_wrapper<F>>(std::move(f))) {}
explicit operator bool() const { return impl != nullptr; }
Ret operator()(Args... args) {
if (!impl) throw std::bad_function_call();
return impl->call(std::forward<Args>(args)...);
}
};
这里的关键是把“调用”抽象成虚函数 call()。不同可调用对象被封装进 callable_wrapper<f></f>,各自实现自己的调用逻辑。外部只看到统一的 operator() 接口。
性能与优化考虑
上述实现使用虚函数调用,有间接跳转开销。真实标准库实现会做更多优化:
- 小对象优化(Small Buffer Optimization):对于小型可调用对象(如普通函数指针、小型 lambda),直接存在对象内部,避免堆分配
- 函数指针代替虚表:有些实现用函数指针数组代替虚函数,减少虚表查找成本
-
内联存储:像 libstdc++ 和 libc++ 都会在
std::function内部预留一段空间存放小闭包
例如,如果 lambda 没有捕获或只捕获少量数据,就不用动态分配内存,提升性能。
总结
C++ 中的类型擦除本质是“用运行时多态模拟泛型能力”。虽然牺牲了一点性能,但换来了极大的灵活性。std::any 和 std::function 正是这一思想的典型应用。
它们的共同特征是:
- 对外提供统一接口
- 内部用模板适配各种类型
- 通过多态或函数指针实现动态分发
- 管理对象生命周期(拷贝、移动、销毁)
理解类型擦除有助于深入掌握现代 C++ 库的设计哲学。基本上就这些。
以上就是c++++如何实现类型擦除(Type Erasure)_c++ std::any与std::function原理解析的详细内容,更多请关注其它相关文章!
# 装进
# seo计费好用吗
# 网站策划建设推荐
# 沁阳关键词排名
# 南京seo公司都选火星
# 网站建设陈洋
# 苏州常规网站建设价格实惠
# 网站自动提交seo
# 小红书如何做营销号推广
# 白驿镇高端网站建设
# 新网站做什么优化好用些
# c++
# 自己的
# 不同类型
# 文件系统
# 绑定
# 尼克
# 如何实现
# 如何使用
# 多态
# 擦除
相关栏目:
【
科技资讯46185 】
【
网络学院92790 】
相关推荐:
小米Civi 4录制视频过暗_小米Civi 4亮度优化
将HTML Canvas内容转换为可上传的图像文件(File对象)
抖音网页版怎么|直播|_抖音网页版开播操作指南
J*a实现学校排课程序_面向对象结构化项目示例
12306选座怎么选到商务座_12306商务座选择与配置说明
怎么在浏览器上运行HTML文件_浏览器运行HTML文件技巧【技巧】
PostgreSQL海量数据高效导入策略:Python与Django实践指南
漫蛙2在线漫画入口 漫蛙正版漫画网页版直达
使用Python高效删除Word宏并转换DOCM为DOCX格式
响应式CSS Grid布局:优化网格项在小屏幕下的堆叠与宽度适配
冬*霸灯泡不亮怎么办_浴霸取暖灯一盏不亮的灯座清洁修复法
vivo手机参数配置怎么增强信号_vivo手机参数配置信号增强方法
J*aScript中如何高效提取对象指定属性
魅族20怎样在浏览器开无图省流_iPhone魅族20浏览器开无图省流【流量节省】
抖音商城签到领现金是真的吗_抖音商城签到奖励与提现说明
抖音隐秘迷城小游戏入口_ 抖音冒险解谜小游戏秒玩
铁路12306官网网页端快速入口 铁路12306官方首页登录教程
vivo云服务网页版登录 怎么登录vivo云服务网页版
Composer的 "licenses" 命令如何帮助你遵守开源协议_检查项目依赖的许可证合规性
蛙漫官网漫画入口地址_蛙漫在线畅读无广告弹窗
C++如何实现线程池_C++11手动实现一个简单的固定大小线程池
C++ map遍历方法大全_C++ map迭代器使用总结
Tabulator表格日期时间排序问题及自定义解决方案
学习通网页版快速入口 学习通官网网页版直接打开
如何在 Excel Online 和 Google 表格中更改日期格式
台积电1.4nm工艺A14瞄准2028:10年来性能提升80%
支付宝如何设置安全保护_支付宝安全设置的全面教程
没有大陆身份证/银行卡如何实名微信? 亲测有效的几种方法分享
Pandas DataFrame:高效添加条件计算列
Lar*el用户头像管理:实现图片缩放、存储与旧文件安全删除的最佳实践
在J*aScript中复现SciPy的B样条拟合与求值:关键考量
夸克浏览器桌面版同步不了书签怎么处理 夸克浏览器跨设备同步异常解决方案
深入理解字体排版:Adobe光学字偶距与CSS字偶距的差异与实现
J*a TimerTask中HashMap意外清空的深层原因与解决方案
C++指针和引用有什么区别_C++内存管理核心概念深度解析
深入理解Google Cloud Datastore查询:祖先路径与数据一致性
Win10怎么设置静态IP地址 Win10手动配置IP地址步骤【指南】
初次安装JDK时环境变量如何正确配置_J*A_HOME与PATH设置规则讲解
整合Supabase认证与Django模型:跨模式迁移的解决方案
qq游戏网页版直接玩_qq游戏免下载快速入口
俄罗斯浏览器官网直达链接 俄罗斯浏览器最新在线入口导航
机器学习中对数变换预测结果的反向还原
处理动态列数据:J*a ArrayList的正确初始化与字符累加教程
C++编译期如何执行复杂计算_C++模板元编程(TMP)技巧与应用
Win11怎么关闭快速启动_Win11彻底关机设置教程
iCloud登录入口网页版 苹果iCloud官网登录
ArrayList与LinkedList操作复杂度详解:遍历与修改
顺丰国际快递查询 国际件官方查询入口
在Go Martini框架中高效服务动态生成图像的实践指南
Go与Ruby之间实现AES加密互通:CFB模式下的密钥长度匹配策略


2025-12-04
浏览次数:次
返回列表