std::any 只能存储可复制或可移动的类型,不支持 void、抽象类、C 风格数组、引用及无拷贝/移动构造函数的类型;支持 int、std::string 等满足要求的类型。
std::any 只能存储可复制(CopyConstructible)或可移动(MoveConstructible)的类型,不支持 void、抽象类、数组类型(如 int[5])、以及带有删除/私有拷贝构造函数的类型。尝试存 void 或未定义拷贝行为的对象会编译失败;存 std::array 没问题,但存 int[3] 会报错 —— 因为 C 风格数组不是对象类型。
int、std::string、std::vector、自定义 class(只要满足拷贝/移动要求)void、int[]、std::function& (引用不能直接存)、const int&(std::any 存的是值,不是引用)std::any 内部用类型擦除实现,每次构造/赋值都有堆分配开销(小对象可能被优化进内部缓冲,但不可依赖)调用 std::any_cast 前必须确保 any_obj 当前持有的类型就是 T,否则抛出 std::bad_any_cast 异常。不能靠“猜”来 cast —— 必须先用 type() 检查或用指针版本做空安全提取。
std::any data = 42;
if (data.type() == typeid(int)) {
int val = std::any_cast(data); // 安全
}
// 更推荐:用指针形式,失败返回 nullptr
if (auto p = std::any_cast(&data)) {
// p 非空,说明 data 是 double 类型
std::cout << *p << "\n";
} else {
// 类型不匹配,不会崩溃
}
std::any 直接用非指针版 std::any_cast
std::any_cast(&any) 返回 T*,空指针表示类型不匹配 —— 这是运行时类型安全检查的核心手段any.type() 返回 std::type_info&,可用于日志、调试或简单分支判断,但注意 typeid 在多态场景下有局限(不带 RTTI 的编译会禁用)std::any 是“任意类型”,运行时完全动态;std::variant 是“有限枚举类型”,编译期就限定可选集合。二者解决的问题不同:std::any 适合插件系统、配置解析等真正无法预知类型的场景;std::variant 更适合状态机、AST 节点、协议字段等已知几种可能的场合。
std::any:无编译期类型约束,体积小(通常 16–32 字节),但每次访问都要运行时检查 + 动态分发std::variant:编译期确定所有可能类型,访问快(无虚函数/RTTI 开销),但不支持新增类型(改 variant 定义就得重编译)std::variant;需要接收用户传入的任意 std::function 或第三方类型?只能用 std::any
在函数接口中滥用 std::any 会严重削弱类型安全和可维护性。比如写一个 process(std::any input),调用方根本不知道该传什么,维护者也无法静态推导行为。它不该出现在 public API 的参数或返回值中,而应作为内部缓存、配置表、或反射机制的底层载体。
std::any parse_config(const std::string& key) —— 应该返回具体结构体或用 std::optional + 明确模板特化std::any 指向工厂函数;序列化器内部用 std::any 缓存临时解析结果std::any 对象生命周期结束,它持有的资源(如 std::string)自动析构 —— 但若你用 std::any_cast 取了内部地址并长期持有,就会悬垂std::any 的“动态”是以运行时开销和显式检查为代价换来的。最容易被忽略的,是忘记用指针版 any_cast 做安全解包,或者在本该用 std::variant 的地方强行上 std::any —— 这两类错
误在调试阶段往往只暴露为崩溃或静默数据截断。