0

0

c++中的标签联合体(Tagged Union)是什么_c++ std::variant实现原理

穿越時空

穿越時空

发布时间:2025-12-04 14:22:02

|

891人浏览过

|

来源于php中文网

原创

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

c++中的标签联合体(tagged union)是什么_c++ std::variant实现原理

标签联合体(Tagged Union)是一种能存储多种不同类型数据,但每次只保存其中一种的数据结构。它和普通联合体(union)的关键区别在于:标签联合体自带一个“标签”(tag),用来标识当前存储的是哪种类型。这使得在读取数据时可以安全判断类型,避免误读导致未定义行为。

C++17 引入的 std::variant 就是标签联合体的标准实现。它提供类型安全的多态存储能力,相比传统的 union 更加安全、易用。

std::variant 的基本用法

std::variant 可以持有其模板参数中列出的任意一种类型:

#include 
#include 

int main() {
    std::variant v;
    v = 42;                    // 存 int
    std::cout << std::get(v) << '\n';

    v = 3.14;                  // 存 double
    if (std::holds_alternative(v)) {
        std::cout << std::get(v) << '\n';
    }
}

上面代码展示了赋值、类型判断和取值的基本操作。如果尝试用错误的类型获取值(如对存 double 的 variant 调用 get),会抛出 std::bad_variant_access 异常(前提是使用访问函数而非指针形式)。

立即学习C++免费学习笔记(深入)”;

std::variant 实现原理

std::variant 的内部实现依赖几个关键技术点:

  • 联合体(union)存储实际数据:variant 内部使用一个 union 来存放所有可能类型的实例。union 的大小由最大类型决定,确保能容纳任意成员。
  • 标签(tag)记录当前类型:一个整型变量(通常是 size_t)记录当前 active 的类型索引。例如,variant 中,0 表示 int,1 表示 double。
  • 构造与析构管理:由于 union 不能自动调用构造函数和析构函数,variant 必须手动处理。当赋新值时,先调用旧对象的析构函数(如有),再在 union 内存位置构造新对象(placement new)。
  • 异常安全性:赋值过程中若构造抛出异常,必须保证 variant 进入合法状态(通常是处于 “valueless_by_exception” 状态)。

简化版实现示意:

薏米AI
薏米AI

YMI.AI-快捷、高效的人工智能创作平台

下载
template 
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(空状态占位)等特性,实现更复杂。

与 union 和继承体系的对比

相比传统 C 风格 union,std::variant 安全得多。传统 union 没有标签,程序员需自行管理类型状态,极易出错。

相比基类指针 + 继承的多态方案,std::variant 是值语义,无堆分配、无虚函数开销,性能更高,且避免内存泄漏风险。

它适合用于表达“一个值可能是几种类型之一”的场景,比如解析 JSON、AST 节点、状态机返回值等。

基本上就这些。std::variant 是现代 C++ 类型安全的重要工具之一,其实现巧妙结合了 union、RAII 和模板元编程,既高效又安全。

相关专题

更多
json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

412

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

533

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

309

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

74

2025.09.10

java多态详细介绍
java多态详细介绍

本专题整合了java多态相关内容,阅读专题下面的文章了解更多详细内容。

15

2025.11.27

c语言union的用法
c语言union的用法

c语言union的用法是一种特殊的数据类型,它允许在相同的内存位置存储不同的数据类型,union的使用可以帮助我们节省内存空间,并且可以方便地在不同的数据类型之间进行转换。使用union时需要注意对应的成员是有效的,并且只能同时访问一个成员。本专题为大家提供union相关的文章、下载、课程内容,供大家免费下载体验。

123

2023.09.27

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

318

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

538

2024.08.29

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

43

2026.01.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8.3万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号