std::any是c++17引入的类型安全容器,可存储任意可复制类型,需通过std::any_cast安全提取值,并支持运行时类型检查,适用于灵活类型场景但存在性能开销。

在C++17中,std::any 被引入到标准库中,它允许你持有任意类型的单一值。这在需要类型擦除但又不想使用void指针或继承时非常有用。相比 void*,std::any 是类型安全的,能自动管理对象的生命周期。
基本用法:存储和访问任意类型
要使用 std::any,需要包含头文件 <any>。你可以用它保存 int、String、自定义类等任何可复制的类型。
示例:
#include <any>
#include <iostream>
#include <string>
std::any value = 42; // 存储 int
value = std::string{“Hello”}; // 替换为 string
value = 3.14; // 再替换为 double
安全地提取值:std::any_cast
从 std::any 中取出值必须使用 std::any_cast,否则会引发异常或返回空指针(取决于用法)。
立即学习“C++免费学习笔记(深入)”;
有两种常用方式:
- 如果确定类型,直接用 any_cast<Type>(any_obj) 获取值,若类型不匹配会抛出 std::bad_any_cast 异常。
- 使用指针形式 any_cast<Type>(&any_obj),失败时返回 nullptr,适合做类型检查。
示例:
try {
int n = std::any_cast
std::cout << “Value is int: ” << n << “n”;
} catch (const std::bad_any_cast&) {
std::cout << “Value is not an intn”;
}
// 安全检查方式
if (auto str = std::any_cast<:string>(&value)) {
std::cout << “got string: ” << *str << “n”;
}
检查当前存储的类型
可以使用 .type() 方法获取当前 any 对象所存值的类型信息,返回 const std::type_info&,常用于调试或运行时判断。
示例:
if (value.type() == typeid(int)) {
std::cout << “Currently holds an intn”;
} else if (value.type() == typeid(std::string)) {
std::cout << “Currently holds a stringn”;
}
注意事项和限制
虽然 std::any 很灵活,但也有一些需要注意的地方:
- 性能开销:类型信息保存和动态调度带来一定运行时成本,不适合高频调用场景。
- 只支持可复制类型:不能放入不可复制的对象(如 unique_ptr 可以,但需注意所有权)。
- 不支持多值:一个 std::any 只能持有一个值,多个值建议配合容器如 vector<any> 使用。
- 空状态:未初始化或被 reset 后处于空状态,此时 any_cast 会失败。
清空 any 的方法:
value.reset(); // 清空内容,变为无类型状态
if (!value.has_value()) {
std::cout << “No value storedn”;
}
基本上就这些。std::any 提供了一种轻量级的类型擦除机制,适用于配置、参数传递、插件系统等需要灵活类型的场合,但别滥用,保持类型明确仍是C++的最佳实践。


