0

0

c++如何使用variant类型_c++17安全的union【技巧】

尼克

尼克

发布时间:2026-02-18 17:02:57

|

656人浏览过

|

来源于php中文网

原创

std::variant 是 c++17 引入的类型安全联合体,本质是“只能持有一种类型的容器”,通过自动管理构造/析构/赋值并强制类型检查访问(如 std::get、std::visit)避免裸 union 的未定义行为。

c++如何使用variant类型_c++17安全的union【技巧】

std::variant 是什么,为什么比裸 union 安全

std::variant 是 C++17 引入的类型安全联合体,本质是一个“只能持有一种类型的容器”。它内部自动管理当前活跃成员的构造、析构和赋值,彻底规避裸 union 手动调用构造函数/析构函数的错误风险。比如你写 union { std::string s; int i; };,直接赋值 s = "hello" 后再读 i —— 这是未定义行为;而 std::variant<:string int></:string> 强制你通过 std::getstd::visit 访问,且访问前会检查当前持有类型。

如何正确初始化和修改 std::variant

初始化必须明确指定哪个分支被激活,不能留空:

  • 直接初始化:std::variant<int std::string> v = 42;</int> → 持有 int
  • 使用 std::in_place_type_t 显式构造(尤其对非默认构造类型):std::variant<:string double> v(std::in_place_type<:string>, "hi");</:string></:string>
  • 赋值时会自动销毁旧值、构造新值:v = std::string("new"); 是安全的;但 v = 3.14; 会先析构 string,再构造 double
  • 禁止隐式转换到歧义类型:若 std::variant<int long></int>,写 v = 5 编译失败(二者都可接受),必须显式转成 int(5)long(5)

访问 variant 的三种方式及陷阱

不能像裸 union 那样“强行 reinterpret”,必须走类型检查路径:

Unreal Images
Unreal Images

免费的AI图片库

下载
  • std::get<t>(v)</t>:要求当前确实持有 T,否则抛 std::bad_variant_access。适合已知状态的场景,比如刚赋过值后立刻取
  • std::get_if<t>(&v)</t>:返回 T*,若不匹配则返回 nullptr,适合需要判空的逻辑
  • std::visit([](auto&& x) { ... }, v):最推荐,类型安全且支持所有分支统一处理;注意 lambda 参数必须能覆盖 variant 所有类型,否则编译失败
  • 常见错误:对空 variant(如 std::monostate 开头)直接 std::get;或 visit 中漏掉某个类型分支,导致编译不过

性能与兼容性注意事项

std::variant 不是零开销抽象:

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

  • 大小至少为最大分支 size + 对齐 padding + 1 字节 type index(通常 1–8 字节,取决于分支数)
  • 每次赋值/构造都有分支判断和可能的析构/构造开销,高频小对象场景(如每帧百万次)需实测对比裸 union
  • 不能用于 POD 类型集合(如 std::variant<int float></int> 是 POD,但含 std::string 就不是),影响 memcpy 和静态初始化
  • MSVC 19.14+、GCC 7.2+、Clang 5.0+ 支持完整特性;老编译器可能缺 std::visit SFINAE 友好实现

真正麻烦的是嵌套 variant 或和 std::optional 混用时的状态组合爆炸——这时候别硬扛,该用 class 封装就封装,variant 的价值在清晰表达“互斥有限选项”,不在模拟任意复杂状态机。

相关文章

c++速学教程(入门到精通)
c++速学教程(入门到精通)

c++怎么学习?c++怎么入门?c++在哪学?c++怎么学才快?不用担心,这里为大家提供了c++速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

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

770

2023.08.02

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

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

128

2023.09.27

string转int
string转int

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

770

2023.08.02

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

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

572

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

254

2025.08.29

C++中int的含义
C++中int的含义

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

210

2025.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

254

2025.08.29

C++中int、float和double的区别
C++中int、float和double的区别

本专题整合了c++中int和double的区别,阅读专题下面的文章了解更多详细内容。

105

2025.10.23

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

561

2026.02.13

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 9.7万人学习

C 教程
C 教程

共75课时 | 4.8万人学习

C++教程
C++教程

共115课时 | 18.3万人学习

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

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