0

0

C++如何使用std::any存储任意类型数据

P粉602998670

P粉602998670

发布时间:2025-09-09 09:56:01

|

252人浏览过

|

来源于php中文网

原创

std::any通过类型擦除和运行时检查实现任意类型安全存储,区别于void*和union的手动类型管理及std::variant的编译时类型限定,适用于配置系统、插件架构等动态场景,但需注意堆分配和类型检查带来的性能开销,优先使用std::variant或具体类型以提升性能。

c++如何使用std::any存储任意类型数据

在C++中,

std::any
提供了一种类型安全的方式来存储任意类型的值,有点像一个可以动态持有任何类型数据的“盒子”。它允许你在运行时将不同类型的数据赋值给同一个
std::any
对象,并在需要时通过类型安全的机制将其取出。

解决方案

使用

std::any
存储任意类型的数据,核心在于包含
头文件,然后创建
std::any
对象,并利用赋值操作存储数据,最后通过
std::any_cast
来安全地提取数据。

这东西用起来其实挺直观的。你先得

include 
。然后,声明一个
std::any
类型的变量,就像这样:
std::any my_data;

当你需要存储一个值时,直接赋值就行了,它会自动处理类型擦除:

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

#include 
#include 
#include 
#include 

int main() {
    std::any my_data; // 创建一个空的any对象

    my_data = 42; // 存储一个int
    std::cout << "存储了int: " << std::any_cast(my_data) << std::endl;

    my_data = std::string("Hello, std::any!"); // 存储一个std::string
    std::cout << "存储了string: " << std::any_cast(my_data) << std::endl;

    my_data = std::vector{1.1, 2.2, 3.3}; // 存储一个std::vector
    // 取出并打印vector的第一个元素
    std::cout << "存储了vector,第一个元素是: " << std::any_cast>(my_data)[0] << std::endl;

    // 尝试取出不存在的类型会抛出std::bad_any_cast异常
    try {
        std::cout << std::any_cast(my_data) << std::endl;
    } catch (const std::bad_any_cast& e) {
        std::cerr << "捕获到异常: " << e.what() << std::endl;
    }

    // 检查是否有值,以及当前存储的类型
    if (my_data.has_value()) {
        std::cout << "my_data当前有值,类型是: " << my_data.type().name() << std::endl;
    }

    my_data.reset(); // 清空any对象
    if (!my_data.has_value()) {
        std::cout << "my_data已经被清空。" << std::endl;
    }

    return 0;
}

这段代码展示了

std::any
的基本操作:赋值、
any_cast
取值、异常处理,以及
has_value()
type()
的使用。
any_cast
是这里的关键,它不仅取回值,还会在类型不匹配时抛出
std::bad_any_cast
异常,这保证了类型安全性。

std::any与void*、union、std::variant有何本质区别

这个问题问得好,因为很多时候我们看到

std::any
都会联想到那些老朋友。但它们之间其实有着非常根本的区别,理解这些差异能帮助我们更好地选择工具

在我看来,

std::any
最核心的价值在于它在运行时提供了类型安全的任意类型存储能力,并且能处理任意复杂类型

void*
是最原始的“万能指针”,它能指向任何类型的数据。但问题在于,
void*
本身不带任何类型信息,你必须手动记住它指向的是什么,然后强制转换回去。一旦转换错了,程序行为就未定义了,这简直就是个定时炸弹,编译期根本发现不了。
std::any
则不然,它内部维护了类型信息,
any_cast
会在运行时检查类型,不匹配就报错,这安全性高了不止一个档次。

union
嘛,它在C语言时代就有了,主要用于在同一块内存空间存储不同类型的数据,但同一时间只能有一种类型是有效的。它的限制很多,比如成员类型通常得是POD类型(Plain Old Data),不能有复杂的构造函数、析构函数。你得自己管理它的生命周期,而且同样没有内置的类型安全检查。
std::any
可以存储任何带有构造函数、析构函数的复杂C++对象,并且它自己会管理这些对象的生命周期,比如在赋值新值时正确销毁旧值。

std::variant
,这是C++17引入的另一个利器,它和
std::any
有些相似,都是类型安全的“和类型”(sum type)。但
std::variant
的关键在于,它能存储的类型集合必须在编译时就确定。比如
std::variant
,它只能是这三种类型之一。它的优点是通常没有堆内存分配(除非内部类型本身需要),性能通常更好,而且编译期就能进行类型检查。
std::any
则没有这个限制,它可以在运行时存储任何类型,你不需要提前知道所有可能的类型。所以,如果你的类型集合是固定的,并且你知道它们是什么,
std::variant
往往是更优的选择;如果类型是完全动态、不可预测的,
std::any
就派上用场了。

说白了,

void*
union
是“手动挡”,效率可能高但风险大;
std::variant
是“自动挡”,但车型(类型)是固定的;
std::any
则是“智能电动车”,能适应各种路况(类型),但可能需要一些额外的“电量”(运行时开销)。

NetShop网店系统
NetShop网店系统

NetShop软件特点介绍: 1、使用ASP.Net(c#)2.0、多层结构开发 2、前台设计不采用任何.NET内置控件读取数据,完全标签化模板处理,加快读取速度3、安全的数据添加删除读取操作,利用存储过程模式彻底防制SQL注入式攻击4、前台架构DIV+CSS兼容IE6,IE7,FF等,有利于搜索引挚收录5、后台内置强大的功能,整合多家网店系统的功能,加以优化。6、支持三种类型的数据库:Acces

下载

在实际项目中,何时应该考虑使用std::any?

std::any
虽然强大,但它不是万能药,也不是应该随处可见的工具。它的最佳应用场景通常集中在那些需要高度灵活性,且类型在编译期难以确定的地方。

我个人觉得,最典型的应用场景就是配置系统。想象一下,你的应用程序需要从配置文件(比如JSON、YAML)中读取各种设置,有些是整数,有些是字符串,有些是布尔值,甚至可能是更复杂的自定义对象。如果用一堆

if-else
switch
来判断类型并存储,代码会变得非常臃肿。这时,你可以用
std::map
来存储配置项,键是配置名,值就是对应的任意类型数据。读取时,根据配置名取出
std::any
对象,再尝试
any_cast
到预期类型,如果失败就说明配置错误或者类型不匹配,这样处理起来既灵活又健壮。

另一个我能想到的地方是插件系统或扩展架构。当你的主程序需要加载外部插件,而这些插件可能会返回各种不同类型的数据时,

std::any
就很有用。主程序不需要知道每个插件具体返回什么类型,只需要约定一个接口,让插件返回
std::any
,然后主程序根据上下文尝试解析。这提供了一种松散耦合的机制。

事件系统中的事件数据载荷也是一个好例子。不同的事件可能携带不同结构的数据。一个

Event
基类可能有一个
std::any payload;
成员,
MouseClickEvent
把它设为
MouseCoords
结构体,
KeyboardEvent
把它设为
KeyCode
。事件处理器在接收到事件后,根据事件类型再决定如何
any_cast
这个
payload

当然,也要警惕过度使用。如果你的数据类型是固定的几个,或者你可以通过多态(继承)来解决,那么

std::any
可能就不是最好的选择。它引入的运行时开销和类型擦除,可能会让代码的意图变得不那么直接,甚至增加了调试难度。所以,用之前最好先问问自己:真的有必要这么灵活吗?有没有更直接、编译期更安全的方案?

使用std::any时可能遇到的性能问题及最佳实践是什么?

std::any
的便利性并非没有代价,性能问题是我们在使用时需要特别留意的。它在实现上,为了能够存储任意类型,通常会利用小对象优化(Small Object Optimization, SOO)

简单来说,如果存储的类型足够小(比如

int
char
等,通常小于
std::any
内部预留的固定大小缓冲区,这个大小一般是16到32字节),那么数据会直接存储在
std::any
对象的内部。这种情况下,性能开销相对较小,没有堆内存分配。但一旦你存储的数据类型超过了这个内部缓冲区的大小(比如一个
std::string
包含长文本,或者一个
std::vector
),
std::any
就会在堆上动态分配内存来存储你的数据。频繁的堆分配和释放是众所周知的性能杀手,它可能导致缓存局部性变差,增加内存碎片,并带来额外的开销。

此外,每次使用

std::any_cast
取值时,都会有运行时类型检查的开销。虽然这通常很快,但在极度性能敏感的循环中,累积起来也可能成为瓶颈。类型擦除本身也意味着一些间接调用(通过函数指针或虚函数),这比直接调用要慢一些。

基于这些考量,我总结了一些使用

std::any
的最佳实践:

  1. 优先考虑
    std::variant
    这点我之前提过,如果你的类型集合是已知的且有限的,
    std::variant
    几乎总是比
    std::any
    更好的选择。它通常没有堆分配,类型安全检查在编译期就能完成,性能和可读性都更优。
  2. 最小化
    any_cast
    的次数:
    尽量在获取到具体类型后,就用该具体类型的变量进行操作,而不是反复从
    std::any
    any_cast
    。比如,先
    auto& val = std::any_cast(my_any);
    ,然后对
    val
    进行一系列操作。
  3. 避免在热点路径(性能关键代码)中频繁使用
    std::any
    如果你的代码需要处理大量数据,并且对性能要求极高,那么
    std::any
    可能不是一个好选择。它的动态特性和潜在的堆分配,可能会让性能难以预测。
  4. 注意对象的生命周期和所有权:
    std::any
    内部存储的是值的副本,或者通过移动语义获取所有权。这意味着你传入的对象会被复制或移动。对于大对象,这会增加开销。如果你想存储指针或引用,确保被指向或引用的对象生命周期足够长。
  5. std::any
    存储的类型提供移动构造函数:
    如果你的类型支持移动语义,
    std::any
    在存储和赋值时会优先使用移动构造而不是拷贝构造,这对于减少开销非常有帮助,尤其是在处理大对象时。
  6. 文档化预期类型: 由于
    std::any
    抹去了类型信息,代码的读者很难一眼看出
    std::any
    变量在特定上下文中应该存储什么类型。因此,清晰的注释或文档变得尤为重要,说明
    std::any
    在某个点上可能包含哪些类型,以及如何安全地提取它们。

总之,

std::any
是一个强大的工具,但它更像是C++工具箱里的“瑞士军刀”,适合解决那些用其他工具不那么优雅、甚至解决不了的特定问题。在日常开发中,我们还是应该优先考虑更具类型安全和性能优势的传统C++构造。

相关专题

更多
C语言变量命名
C语言变量命名

c语言变量名规则是:1、变量名以英文字母开头;2、变量名中的字母是区分大小写的;3、变量名不能是关键字;4、变量名中不能包含空格、标点符号和类型说明符。php中文网还提供c语言变量的相关下载、相关课程等内容,供大家免费下载使用。

379

2023.06.20

c语言入门自学零基础
c语言入门自学零基础

C语言是当代人学习及生活中的必备基础知识,应用十分广泛,本专题为大家c语言入门自学零基础的相关文章,以及相关课程,感兴趣的朋友千万不要错过了。

608

2023.07.25

c语言运算符的优先级顺序
c语言运算符的优先级顺序

c语言运算符的优先级顺序是括号运算符 > 一元运算符 > 算术运算符 > 移位运算符 > 关系运算符 > 位运算符 > 逻辑运算符 > 赋值运算符 > 逗号运算符。本专题为大家提供c语言运算符相关的各种文章、以及下载和课程。

348

2023.08.02

c语言数据结构
c语言数据结构

数据结构是指将数据按照一定的方式组织和存储的方法。它是计算机科学中的重要概念,用来描述和解决实际问题中的数据组织和处理问题。数据结构可以分为线性结构和非线性结构。线性结构包括数组、链表、堆栈和队列等,而非线性结构包括树和图等。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

255

2023.08.09

c语言random函数用法
c语言random函数用法

c语言random函数用法:1、random.random,随机生成(0,1)之间的浮点数;2、random.randint,随机生成在范围之内的整数,两个参数分别表示上限和下限;3、random.randrange,在指定范围内,按指定基数递增的集合中获得一个随机数;4、random.choice,从序列中随机抽选一个数;5、random.shuffle,随机排序。

585

2023.09.05

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

519

2023.09.20

c语言get函数的用法
c语言get函数的用法

get函数是一个用于从输入流中获取字符的函数。可以从键盘、文件或其他输入设备中读取字符,并将其存储在指定的变量中。本文介绍了get函数的用法以及一些相关的注意事项。希望这篇文章能够帮助你更好地理解和使用get函数 。

632

2023.09.20

c数组初始化的方法
c数组初始化的方法

c语言数组初始化的方法有直接赋值法、不完全初始化法、省略数组长度法和二维数组初始化法。详细介绍:1、直接赋值法,这种方法可以直接将数组的值进行初始化;2、不完全初始化法,。这种方法可以在一定程度上节省内存空间;3、省略数组长度法,这种方法可以让编译器自动计算数组的长度;4、二维数组初始化法等等。

595

2023.09.22

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

74

2025.12.31

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8.1万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.1万人学习

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

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