std::any支持任意类型存储,适用于类型未知的场景,但性能开销大;std::variant限定于预定义类型集合,类型安全且高效,推荐在类型明确时使用以提升性能和可读性。

在c++17中,std::any 和 std::variant 都是用于处理多种类型的数据容器,但它们的设计目标和使用场景有显著区别。理解它们的差异有助于在实际开发中做出合适选择。
std::any:任意类型的容器
std::any 可以保存任意类型的值,是一种类型安全的“万能”容器。它适用于你无法预知将要存储什么类型的场景。
特点:
常见使用场景:
立即学习“C++免费学习笔记(深入)”;
基本用法示例:
#include <any> #include <iostream> #include <string> std::any data = 42; data = std::string("hello"); if (data.type() == typeid(std::string)) { std::cout << std::any_cast<std::string>(data); }
std::variant:有限类型的联合体
std::variant 是一个类型安全的联合体(union),只能保存预先指定的几种类型之一。它是“要么是A,要么是B,要么是C”这种逻辑的理想选择。
特点:
- 类型集合在编译期确定
- 内存大小为最大可能类型的尺寸,通常在栈上分配
- 支持访问者模式(std::visit)进行类型分发
- 性能优于 std::any,没有动态分配开销
常见使用场景:
立即学习“C++免费学习笔记(深入)”;
基本用法示例:
#include <variant> #include <iostream> #include <string> std::variant<int, std::string, bool> v = "text"; v = 123; std::visit([](auto& val) { std::cout << val << 'n'; }, v);
关键区别总结
两者最核心的区别在于类型约束方式和性能特性:
- std::any 支持任意类型,灵活但慢;std::variant 类型固定,高效且安全
- std::any 使用 type() 和 any_cast 查询和提取;std::variant 推荐用 std::visit 或 std::get 进行类型处理
- std::variant 可以避免空状态(除非显式包含 std::monostate),而 std::any 初始为空
如何选择?
如果你需要一个能存“任何东西”的容器,比如构建通用容器或中间数据传递,用 std::any。但大多数情况下,当你知道所有可能的类型时,优先选择 std::variant,因为它更安全、更快、更清晰地表达了设计意图。
基本上就这些。合理使用这两个工具,可以让现代C++代码更灵活又不失类型安全。


