应使用 std::holds_alternative、std::get_if 或 std::visit 安全访问 std::variant;避免直接 std::get 导致异常,优先用 if constexpr + std::visit 实现编译期分发,勿将其用于运行时多态场景。
直接调用 std::get 前必须确保 v 当前持有类型 T,否则运行时抛出 std::bad_variant_access。这不是设计缺陷,而是类型安全的强制约束。
std::holds_alternative(v) 先检查再取值,适合分支逻辑明确的场景std::get_if(&v) 获取指针,空指针表示不匹配,适合需要原地修改或避免异常的上下文std::holds_alternative + std::get,可改用 std::visit 一次性分发不要手写 visitor 类;C++17 起支持闭包作为 visitor,配合 auto 参数推导可自动匹配各分支,代码紧凑且类型安全。
std::variantv = "hello"; std::visit([](const auto& x) { using T = std::decay_t ; if constexpr (std::is_same_v ) { std::cout << "int: " << x << "\n"; } else if constexpr (std::is_same_v ) { std::cout << "string: " << x << "\n"; } else if constexpr (std::is_same_v ) { std::cout << "double: " << x << "\n"; } }, v);
if constexpr 是关键:编译期剔除不匹配分支,避免模板实例化错误const auto& 或 auto&&,否则无法绑定所有备选类型std::v 是编译期确定类型的“静态多态”,和
ariantvirtual、dynamic_cast 完全无关。试图把它当基类指针容器用,说明模型设计已偏离其定位。
std::unique_ptr 而非 std::variant
std::variant<:variant>, ...> 是反模式;应扁平化为单层 variant 或用递归变参模板封装std::variant 的 move 构造和 move 赋值会转移当前所含对象的资源,但前提是该类型满足可移动——否则退化为拷贝,甚至编译失败。
std::mutex、引用成员)的类型不能放进 std::variant
std::move(v) 后,v 进入有效但未指定状态(通常为第一个备选类型的默认构造值),不可再直接访问v.emplace<:string>("abc"))会就地构造,避免临时对象拷贝,比 v = std::string("abc") 更高效实际用好 std::variant 的关键,是接受它“一个变量只能持有一个确定类型”的本质——不是万能多态容器,而是类型安全的单槽存储。越早放弃模拟运行时多态的念头,越容易写出清晰、可维护的代码。