0

0

C++ variant怎么用 C++17类型安全联合体使用指南【泛型】

尼克

尼克

发布时间:2026-02-01 17:41:02

|

155人浏览过

|

来源于php中文网

原创

std::variant 是 C++17 引入的类型安全联合体,区别于 union:它自动管理构造/析构、支持运行时类型查询与安全访问(如 std::visit、std::get),要求备选类型可析构和可移动,禁止数组/引用/void 等类型。

c++ variant怎么用 c++17类型安全联合体使用指南【泛型】

std::variant 是什么,和 union 有什么区别

std::variant 是 C++17 引入的类型安全联合体,本质是一个“只能持有一种类型的容器”,编译期就限定可选类型集合,运行时能安全知道当前存的是哪个。它不是 union 的语法糖——原生 union 不调用构造/析构、不检查访问类型、容易 UB;而 std::variant 自动管理对象生命周期,提供 std::visitstd::get 等安全访问机制。

常见错误现象:直接对 std::variant 成员取地址或强制 reinterpret_cast;试图像 union 那样“覆盖写入”不同分支;忽略 valueless_by_exception 状态。

  • 必须显式初始化(默认构造只对第一个可默认构造的类型生效)
  • 所有备选类型必须满足可析构、可移动(部分场景还需可复制)
  • 不支持数组、引用、void、某些不完整类型

怎么安全获取 variant 中的值:get vs visit

std::get(v) 直接按类型提取,但若当前不持有 T,抛出 std::bad_variant_accessstd::get(v) 按索引提取,越界同样抛异常。适合你确定类型且愿意承担异常成本的场景。

std::visit 是更推荐的方式,尤其在泛型上下文中——它接受一个可调用对象(lambda、函数对象等),编译期生成所有分支的 dispatch 表,运行时根据当前 index 调用对应重载。

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

std::variant v = "hello";
std::visit([](const auto& x) {
    using T = std::decay_t;
    if constexpr (std::is_same_v) {
        std::cout << "int: " << x;
    } else if constexpr (std::is_same_v) {
        std::cout << "string: " << x;
    } else if constexpr (std::is_same_v) {
        std::cout << "double: " << x;
    }
}, v);

注意:

Melodrive
Melodrive

Melodrive -一个AI音乐引擎,根据用户的情绪状态和喜好生成个性化的音乐。

下载
  • lambda 参数必须能匹配所有备选类型(否则编译失败)
  • std::visit 不支持“漏掉某个类型”的写法;若想 fallback,可用 std::holds_alternative + 手动分支
  • 若 variant 处于 valueless_by_exception 状态,std::visit 会抛异常

泛型处理 variant:如何写一个通用的 to_string

写泛型转换函数的关键是避免为每个新类型重复实现,同时保持类型安全。推荐用 std::visit + lambda + 可变参数模板组合:

template
struct to_string_visitor {
    std::string operator()(const Ts& t) const { return std::to_string(t); }
    std::string operator()(const std::string& s) const { return "\"" + s + "\""; }
    std::string operator()(const auto&) const { return "unknown"; }
};

template std::string to_string(const std::variant& v) { return std::visit(to_string_visitor{}, v); }

使用场景:

  • 日志打印、配置序列化、调试输出
  • 需要适配自定义类型时,在 visitor 中特化 operator()

容易踩的坑:

  • 忘记给 visitor 加 const 限定符,导致无法绑定 const variant
  • 在 visitor 中返回类型不一致(比如有的分支返回 std::string,有的返回 void),编译失败
  • std::monostate 或空状态没做处理,访问时报错

性能与兼容性要注意什么

std::variant 的空间开销是所有备选类型中最大 size 加上一个字节(用于存储 index);时间开销主要在 std::visit 的一次查表跳转,基本可视为 O(1),远优于动态类型如 std::any

  • MSVC 19.14+、GCC 7.2+、Clang 5.0+ 完整支持;老版本需谨慎
  • std::variant 不支持 constexpr 构造(C++20 才部分支持)
  • 移动语义是安全的,但拷贝构造要求所有类型都可拷贝;若含不可拷贝类型(如 std::unique_ptr),只能移动

关键提醒:不要为了“省一个字节”把小类型(如 bool)和大类型(如 std::vector)塞进同一个 variant——index 存储本身不贵,但缓存局部性会被破坏,实际性能可能比两个独立变量还差。真正该关心的是访问模式是否集中、分支是否可预测。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
string转int
string转int

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

503

2023.08.02

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

532

2023.09.20

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

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

126

2023.09.27

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

177

2023.11.23

java中void的含义
java中void的含义

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

102

2025.11.27

lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

208

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

191

2025.11.08

Python lambda详解
Python lambda详解

本专题整合了Python lambda函数相关教程,阅读下面的文章了解更多详细内容。

56

2026.01.05

go语言 注释编码
go语言 注释编码

本专题整合了go语言注释、注释规范等等内容,阅读专题下面的文章了解更多详细内容。

30

2026.01.31

热门下载

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

精品课程

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

共61课时 | 3.7万人学习

C# 教程
C# 教程

共94课时 | 8.2万人学习

C 教程
C 教程

共75课时 | 4.4万人学习

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

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