
在C++17中引入的std::variant是一个类型安全的联合体(union),可以保存多种类型中的某一种值。它解决了传统union类型不安全的问题,使用起来更可靠且易于管理。
基本用法
std::variant定义在头文件中。声明一个variant时,需要指定它可以容纳的类型列表。
#include iostream>
例如,创建一个可以存储int、double或std::string的variant:
std::variantv = 42; // 存int
v = 3.14; // 存double
v = "hello"; // 存string
访问variant中的值
不能直接解引用variant,必须通过正确方式获取其当前持有的值。
立即学习“C++免费学习笔记(深入)”;
1. 使用std::get
可以通过类型或索引来获取值,但必须确保类型匹配,否则会抛出std::bad_variant_access异常。
if (std::holds_alternative
double val = std::get
std::cout }
2. 使用std::get_if
返回指针,适合检查并安全访问值。
if (auto* p = std::get_ifstd::cout } else if (auto* p = std::get_if<:string>(&v)) {
std::cout }
判断当前类型
使用std::holds_alternative检查当前存储的类型:
std::cout }
也可以用v.index()获取当前类型的索引(从0开始):
case 0: /* int */ break;
case 1: /* double */ break;
case 2: /* string */ break;
}
结合std::visit进行类型分发
最强大的功能是配合std::visit,实现对不同类型的统一处理。
std::cout }, v);
也可以写成具名lambda或函数对象:
struct Printer {void operator()(int i) const { std::cout void operator()(double d) const { std::cout void operator()(const std::string& s) const { std::cout };
std::visit(Printer{}, v);
基本上就这些。std::variant让多类型值的安全管理变得简单,特别适合解析配置、表达式求值、状态机等场景。注意编译器需支持C++17及以上标准,并开启对应模式(如g++ -std=c++17)。











