0

0

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

尼克

尼克

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

|

175人浏览过

|

来源于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<t>(v)</t> 直接按类型提取,但若当前不持有 T,抛出 std::bad_variant_accessstd::get<i>(v)</i> 按索引提取,越界同样抛异常。适合你确定类型且愿意承担异常成本的场景。

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

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

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

注意:

如此AI员工
如此AI员工

国内首个全链路营销获客AI Agent

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

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

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

template<typename... Ts>
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"; }
};
<p>template<typename... Ts>
std::string to_string(const std::variant<Ts...>& v) {
return std::visit(to_string_visitor<Ts...>{}, v);
}</p>

使用场景:

  • 日志打印、配置序列化、调试输出
  • 需要适配自定义类型时,在 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

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
string转int
string转int

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

1091

2023.08.02

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

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

565

2023.09.20

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

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

129

2023.09.27

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

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

188

2023.11.23

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

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

136

2025.11.27

lambda表达式
lambda表达式

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

215

2023.09.15

python lambda函数
python lambda函数

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

193

2025.11.08

Python lambda详解
Python lambda详解

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

62

2026.01.05

Python WebSocket实时通信与异步服务开发实践
Python WebSocket实时通信与异步服务开发实践

本专题聚焦 Python 在实时通信场景中的开发实践,系统讲解 WebSocket 协议原理、长连接管理、消息推送机制以及异步服务架构设计。内容包括客户端与服务端通信实现、连接稳定性优化、消息队列集成及高并发处理策略。通过完整案例,帮助开发者构建高效稳定的实时通信系统,适用于聊天应用、实时数据推送等场景。

7

2026.03.18

热门下载

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

精品课程

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

共61课时 | 4.4万人学习

C# 教程
C# 教程

共94课时 | 11.6万人学习

C 教程
C 教程

共75课时 | 5.6万人学习

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

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