首页 > 后端开发 > C++ > 正文

C++17的std::variant怎么用_C++类型安全的联合体std::variant访问方法

穿越時空
发布: 2025-12-02 13:09:53
原创
146人浏览过
std::variant是C++17引入的类型安全联合体,可持有多种类型之一,支持赋值、emplace初始化,通过std::get、std::get_if、std::holds_alternative安全访问,推荐使用std::visit进行类型分发,适用于JSON解析等多类型场景。

c++17的std::variant怎么用_c++类型安全的联合体std::variant访问方法

在 C++17 中,std::variant 是一个类型安全的联合体(type-safe union),用来表示可以持有多种类型之一的对象。与传统的 union 不同,std::variant 能知道当前存储的是哪种类型,避免了类型错误访问的问题,大大提升了类型安全性。

基本用法:定义和赋值

你可以将 std::variant 看作一个能“装”多个类型的容器,但每次只能保存其中一个类型的一个值。

示例:

#include <variant>
#include <iostream>
<p>int main() {
std::variant<int, double, std::string> v;</p><pre class='brush:php;toolbar:false;'>v = 42;           // 持有 int
v = 3.14;         // 持有 double
v = "hello";      // 持有 std::string

return 0;
登录后复制

}

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

初始化方式支持直接赋值、构造函数、emplace 等:

  • std::variant<int, std::string> v = "hello";
  • v.emplace<1>("world"); // 按索引构造
  • v.emplace<std::string>("hi"); // 按类型构造

如何安全地访问 variant 中的值

由于 variant 可能包含不同类型,直接访问容易出错。C++17 提供了多种安全访问方法:

Otter.ai
Otter.ai

一个自动的会议记录和笔记工具,会议内容生成和实时转录

Otter.ai 91
查看详情 Otter.ai

1. std::get<T>(v) 或 std::get<I>(v)

通过类型或索引获取值,但如果类型不匹配会抛出 std::bad_variant_access 异常。

try {
    std::cout << std::get<double>(v) << '\n';
} catch (const std::bad_variant_access&) {
    std::cout << "当前不是 double 类型!\n";
}
登录后复制

2. std::get_if<T>(&v)

返回指向当前值的指针,如果类型不匹配则返回 nullptr。适合判断和访问同时进行。

if (auto* p = std::get_if<int>(&v)) {
    std::cout << "int 值为: " << *p << '\n';
} else if (auto* p = std::get_if<std::string>(&v)) {
    std::cout << "字符串: " << *p << '\n';
}
登录后复制

3. std::holds_alternative<T>(v)

判断当前 variant 是否持有指定类型。

if (std::holds_alternative<double>(v)) {
    std::cout << "当前是 double: " << std::get<double>(v) << '\n';
}
登录后复制

使用 std::visit 进行类型分发(推荐)

最强大且通用的方式是 std::visit,它能对 variant 当前持有的值应用一个可调用对象(如 lambda),自动匹配类型。

std::visit([](const auto& value) {
    std::cout << "值是: " << value << ",类型为: " 
              << typeid(value).name() << '\n';
}, v);
登录后复制

也可以使用多个 lambda 处理不同逻辑:

std::visit([<auto&>(const auto& x) {
    using T = std::decay_t<decltype(x)>;
    if constexpr (std::is_same_v<T, int>)
        std::cout << "整数: " << x << '\n';
    else if constexpr (std::is_same_v<T, double>)
        std::cout << "浮点: " << x << '\n';
    else if constexpr (std::is_same_v<T, std::string>)
        std::cout << "字符串: " << x << '\n';
}, v);
登录后复制

这种方式在编译期生成对应代码,效率高且类型安全。

注意事项和常见技巧

  • 默认构造:variant 默认构造时会构造其第一个类型(前提是该类型可默认构造)。
  • 空状态:如果第一个类型不可默认构造,variant 会处于“未初始化”状态,需手动赋值。
  • 异常安全:赋值或 emplace 失败时,variant 可能进入“无价值”状态,可通过 v.valueless_by_exception() 判断。
  • 性能:variant 的大小等于最大类型的大小加上少量用于类型标识的空间,通常比继承结构更轻量。

基本上就这些。std::variant 非常适合处理“一个变量可能是几种类型之一”的场景,比如解析 JSON、实现表达式求值、状态机等。配合 std::visitstd::get_if,既能保证类型安全,又能写出清晰高效的代码。

以上就是C++17的std::variant怎么用_C++类型安全的联合体std::variant访问方法的详细内容,更多请关注php中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

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