标签联合体通过标签标识当前存储类型,确保安全访问。std::variant是其标准实现,内部用union存数据、tag记类型,并手动管理构造析构,支持异常安全与多类型值语义,相比union更安全,比继承体系更高效。

标签联合体(Tagged Union)是一种能存储多种不同类型数据,但每次只保存其中一种的数据结构。它和普通联合体(union)的关键区别在于:标签联合体自带一个“标签”(tag),用来标识当前存储的是哪种类型。这使得在读取数据时可以安全判断类型,避免误读导致未定义行为。
C++17 引入的 std::variant 就是标签联合体的标准实现。它提供类型安全的多态存储能力,相比传统的 union 更加安全、易用。
std::variant 可以持有其模板参数中列出的任意一种类型:
#include <variant>
#include <iostream>
int main() {
std::variant<int, double, std::string> v;
v = 42; // 存 int
std::cout << std::get<int>(v) << '\n';
v = 3.14; // 存 double
if (std::holds_alternative<double>(v)) {
std::cout << std::get<double>(v) << '\n';
}
}
上面代码展示了赋值、类型判断和取值的基本操作。如果尝试用错误的类型获取值(如对存 double 的 variant 调用 get<int>),会抛出 std::bad_variant_access 异常(前提是使用访问函数而非指针形式)。
立即学习“C++免费学习笔记(深入)”;
std::variant 的内部实现依赖几个关键技术点:
简化版实现示意:
template <typename T1, typename T2>
class simple_variant {
union {
T1 t1;
T2 t2;
};
int tag; // 当前类型索引
public:
simple_variant(const T1& x) : t1(x), tag(0) {}
simple_variant& operator=(const T2& x) {
if (tag == 0) t1.~T1();
new(&t2) T2(x); // placement new
tag = 1;
return *this;
}
~simple_variant() {
if (tag == 0) t1.~T1();
else if (tag == 1) t2.~T2();
}
};
真实 std::variant 支持任意数量类型(通过可变参数模板)、访问器(visitor 模式)、monostate(空状态占位)等特性,实现更复杂。
相比传统 C 风格 union,std::variant 安全得多。传统 union 没有标签,程序员需自行管理类型状态,极易出错。
相比基类指针 + 继承的多态方案,std::variant 是值语义,无堆分配、无虚函数开销,性能更高,且避免内存泄漏风险。
它适合用于表达“一个值可能是几种类型之一”的场景,比如解析 JSON、AST 节点、状态机返回值等。
基本上就这些。std::variant 是现代 C++ 类型安全的重要工具之一,其实现巧妙结合了 union、RAII 和模板元编程,既高效又安全。
以上就是c++++中的标签联合体(Tagged Union)是什么_c++ std::variant实现原理的详细内容,更多请关注php中文网其它相关文章!
c++怎么学习?c++怎么入门?c++在哪学?c++怎么学才快?不用担心,这里为大家提供了c++速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号