0

0

c++ type erasure技巧 c++如何实现std::any和std::function

星降

星降

发布时间:2026-03-18 15:33:00

|

821人浏览过

|

来源于php中文网

原创

<p>std::any 通过 void* 加函数表实现类型擦除,不依赖虚函数;其核心是构造/析构/拷贝/类型查询函数指针组成的结构体,配合小对象优化与严格生命周期管理。</p>

c++ type erasure技巧 c++如何实现std::any和std::function

std::any 的核心是 void* + 类型擦除函数表

它不是靠虚函数继承实现的,而是用一个 struct 存函数指针:构造、析构、拷贝、类型查询。所有操作都通过这个“函数表”跳转,std::any 本身只管存 void* 和这个表指针。

实操建议:

  • 自己写简易版时,别直接裸存 void*,一定要配对管理生命周期——比如用 new 分配的,就得在析构函数里 delete;否则一 move 或赋值就悬空
  • 类型查询靠 std::type_info,但注意:不同编译单元的同名类(尤其模板实例化)可能生成不同 std::type_info 对象,typeid(T) == typeid(U) 不一定稳,优先用 std::any_cast 的运行时检查
  • 小对象优化(SOO)很关键:像 libstdc++ 和 libc++ 都给 std::any 预留了约 24–32 字节内联空间,避免小类型(如 intstd::string_view)堆分配;自己实现时漏掉这点,性能会差一截

std::function 的类型擦除比 std::any 更重

std::function 不仅要擦除目标类型,还要统一调用签名——所有可调用体(函数指针、lambda、std::bind、仿函数)都被转换成“能响应 operator() 并匹配指定参数/返回类型的黑盒”。

常见错误现象:

立即学习C++免费学习笔记(深入)”;

  • 捕获局部变量的 lambda 赋给 std::function 后,原作用域结束,调用时报 segmentation fault——本质是擦除后没做所有权转移,得确保捕获的内容生命周期 ≥ std::function 本身
  • std::function<void></void> 接收返回 int 的函数,编译失败:擦除前就做签名匹配,不兼容的 callable 根本进不来,不是运行时报错
  • 移动后调用已移动的 std::function,触发 std::bad_function_call——因为擦除函数表里“调用函数指针”被置为 nullptr,但这个状态不会自动清零其他字段,内存未必全脏,所以不能靠 memcmp 判空

用 std::any_storage 和 std::function 的 vtable 手写简易版

没必要从零造轮子,但理解它们怎么组织数据,能避开很多误用。

百灵大模型
百灵大模型

蚂蚁集团自研的多模态AI大模型系列

下载

实操建议:

  • 定义擦除结构体时,把函数指针打包进一个 struct,比如:
    struct any_vtable {
        void (*dtor)(void*);
        void (*copy)(void*, const void*);
        const std::type_info& (*type)(void*);
    };
    别把每个函数单独存为成员变量,缓存不友好,也难扩展
  • lambda 擦除进 std::function 时,编译器会生成一个唯一闭包类型,并把捕获数据作为该类型的成员;若捕获为空,有些实现(如 MSVC)会直接退化为函数指针,但别依赖——sizeof(std::function<int>)</int> 在不同 STL 实现里可能差 8~16 字节
  • 别在 std::any 里存多态对象(带虚函数的类),除非你手动保证虚表指针有效;更安全的做法是存智能指针或值语义类型

std::any_cast 和 std::function::target 的陷阱

这两个接口看着像“取出来”,实际都有隐含约束和未定义行为风险。

使用场景与坑点:

  • std::any_cast<t>(&a)</t> 返回 T*,如果 a 里不是 T 类型,返回空指针——但很多人写成 *std::any_cast<t>(&a)</t>,一解引用就崩;应该先判空再用
  • std::function::target<t>()</t> 只能取回原始类型,不能跨继承体系;比如存的是 std::unique_ptr<base>,用 target<:unique_ptr>></:unique_ptr> 拿不到,哪怕 Derived 继承 Base
  • std::functiontarget_type() 返回 std::type_info,但 C++17 起禁止用 == 直接比较两个 std::type_info 对象(标准说“行为未定义”),正确方式是用 std::type_info::hash_code()std::type_info::name()(后者不跨平台)

类型擦除不是魔法,它把编译期信息搬到运行期管理,代价是间接跳转、额外内存、以及所有生命周期必须人工对齐。最常被忽略的,其实是那个“谁负责释放”的契约——擦除层不管,得你自己盯紧。

相关文章

c++速学教程(入门到精通)
c++速学教程(入门到精通)

c++怎么学习?c++怎么入门?c++在哪学?c++怎么学才快?不用担心,这里为大家提供了c++速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

相关标签:

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
function是什么
function是什么

function是函数的意思,是一段具有特定功能的可重复使用的代码块,是程序的基本组成单元之一,可以接受输入参数,执行特定的操作,并返回结果。本专题为大家提供function是什么的相关的文章、下载、课程内容,供大家免费下载体验。

500

2023.08.04

js函数function用法
js函数function用法

js函数function用法有:1、声明函数;2、调用函数;3、函数参数;4、函数返回值;5、匿名函数;6、函数作为参数;7、函数作用域;8、递归函数。本专题提供js函数function用法的相关文章内容,大家可以免费阅读。

167

2023.10.07

Python WebSocket实时通信与异步服务开发实践
Python WebSocket实时通信与异步服务开发实践

本专题聚焦 Python 在实时通信场景中的开发实践,系统讲解 WebSocket 协议原理、长连接管理、消息推送机制以及异步服务架构设计。内容包括客户端与服务端通信实现、连接稳定性优化、消息队列集成及高并发处理策略。通过完整案例,帮助开发者构建高效稳定的实时通信系统,适用于聊天应用、实时数据推送等场景。

2

2026.03.18

Java Spring Security权限控制与认证机制实战
Java Spring Security权限控制与认证机制实战

本专题围绕 Java 后端安全体系建设展开,重点讲解 Spring Security 在权限控制与认证机制中的应用实践。内容涵盖用户认证流程、权限模型设计、JWT 鉴权方案、OAuth2 集成以及接口安全防护策略。通过实际项目案例,帮助开发者构建安全可靠的后端认证体系,提升系统安全性与可扩展能力。

0

2026.03.18

抖漫入口地址合集
抖漫入口地址合集

本专题整合了抖漫入口地址相关合集,阅读专题下面的文章了解更多详细地址。

110

2026.03.17

多环境下的 Nginx 安装、结构与运维实战
多环境下的 Nginx 安装、结构与运维实战

本专题聚焦多环境下Nginx实战,详解开发、测试及生产环境的差异化安装策略与目录结构规划。深入剖析配置模块化设计、灰度发布流程及跨环境同步机制。结合监控告警、故障排查与自动化运维工具,提供全链路管理方案,助力团队构建灵活、高可用的Nginx服务体系,从容应对复杂业务场景挑战。

13

2026.03.17

PS 批量添加图片
PS 批量添加图片

本专题整合了PS批量添加图片教程合集,阅读专题下面的文章了解更多详细操作。

10

2026.03.17

Nginx 基础架构:从安装配置到系统化管理
Nginx 基础架构:从安装配置到系统化管理

本专题深入解析Nginx基础架构,涵盖从源码编译与包管理安装,到核心配置文件优化及虚拟主机部署。进一步探讨日志轮转、性能调优、高可用集群构建及自动化运维策略,助力管理员实现从单一服务搭建到企业级系统化管理的全面升级,确保Web服务高效、稳定运行。

7

2026.03.17

mulerun骡子快跑入口地址汇总
mulerun骡子快跑入口地址汇总

本专题整合了mulerun入口地址合集,阅读专题下面的文章了解更多详细内容。

216

2026.03.17

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 11.5万人学习

C 教程
C 教程

共75课时 | 5.6万人学习

C++教程
C++教程

共115课时 | 22.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号