第 18.15 節

std::variant

0瀏覽次數0訪問次數--跳出率--平均停留

本节解决什么问题

有时候一个变量需要存储"可能是 int,也可能是 string,也可能是 double"的值。传统的做法是 union(C 语言),但它不类型安全——你不知道当前存的是哪种类型,访问错了就崩溃。

std::variant 是类型安全的联合体,能存储多种类型之一,并且知道当前存的是哪种类型

这个特性是什么

std::variant<T1, T2, ...> 是 C++17 引入的类型安全的联合体。在同一时刻,它只存储其中一种类型的值。访问时编译器会帮你检查,不会出现"访问了错误类型"的问题。

C++ 标准版本

C++17

需要的头文件

#include <variant>

基本语法

std::variant<int, double, std::string> v;

v = 42;                      // 存 int
v = 3.14;                    // 存 double
v = std::string("hello");   // 存 string

// 访问方式 1:std::get<T>(v) —— 类型不对抛异常
int n = std::get<int>(v);

// 访问方式 2:std::get_if<T>(&v) —— 类型不对返回 nullptr
if (auto* p = std::get_if<int>(&v)) { ... }

// 访问方式 3:std::visit —— 用 visitor 模式处理所有可能的类型
std::visit([](auto&& val) { ... }, v);

// 查询当前存储的类型的索引
size_t idx = v.index();  // 0-based

常用用法

操作说明
v = value;赋值(自动切换类型)
v.emplace<T>(args...)原地构造
std::get<T>(v)获取值(类型不对抛 std::bad_variant_access
std::get_if<T>(&v)安全获取(类型不对返回 nullptr)
v.index()返回当前类型的索引(0-based)
std::visit(visitor, v)用 visitor 模式处理
std::holds_alternative<T>(v)判断是否持有 T 类型

示例代码

示例 1:variant 基本用法——存不同类型的值

#include <iostream>
#include <variant>
#include <string>
#include <type_traits>

int main()
{
    // v 可以存 int、double 或 string
    std::variant<int, double, std::string> v;

    v = 42;
    std::cout << "int: " << std::get<int>(v) << "\n";

    v = 3.14;
    std::cout << "double: " << std::get<double>(v) << "\n";

    v = std::string("hello");
    std::cout << "string: " << std::get<std::string>(v) << "\n";

    // 查看当前类型索引
    std::cout << "current index: " << v.index() << "\n";  // 2 (string)

    return 0;
}

运行结果

int: 42
double: 3.14
string: hello
current index: 2

示例 2:在示例 1 基础上,用 get_if 安全访问

#include <iostream>
#include <variant>
#include <string>

void print_value(const std::variant<int, double, std::string>& v)
{
    // 安全方式:逐个尝试,get_if 返回指针
    if (auto* p = std::get_if<int>(&v))
    {
        std::cout << "int: " << *p << "\n";
    }
    else if (auto* p = std::get_if<double>(&v))
    {
        std::cout << "double: " << *p << "\n";
    }
    else if (auto* p = std::get_if<std::string>(&v))
    {
        std::cout << "string: " << *p << "\n";
    }
}

int main()
{
    std::variant<int, double, std::string> v;

    v = 42;
    print_value(v);

    v = 3.14159;
    print_value(v);

    v = std::string("C++17");
    print_value(v);

    return 0;
}

运行结果

int: 42
double: 3.14159
string: C++17

示例 3:在示例 2 基础上,用 std::visit 处理所有类型

#include <iostream>
#include <variant>
#include <string>

int main()
{
    std::variant<int, double, std::string> v;

    // std::visit 配合泛型 lambda 优雅处理所有类型
    auto printer = [](const auto& val) {
        std::cout << "value: " << val << "\n";
    };

    v = 42;
    std::visit(printer, v);

    v = 2.718;
    std::visit(printer, v);

    v = std::string("hello variant");
    std::visit(printer, v);

    // 也可以返回不同类型的值
    auto to_double = [](const auto& val) -> double {
        if constexpr (std::is_same_v<std::decay_t<decltype(val)>, std::string>)
        {
            return 0.0;  // string 不能转 double
        }
        else
        {
            return static_cast<double>(val);
        }
    };

    v = 10;
    std::cout << "to_double: " << std::visit(to_double, v) << "\n";

    return 0;
}

运行结果

value: 42
value: 2.718
value: hello variant
to_double: 10

示例 4:在示例 3 基础上,用 variant 表示消息类型

#include <iostream>
#include <variant>
#include <string>

// 定义消息类型
struct TextMessage { std::string text; };
struct NumberMessage { int number; };
struct QuitMessage {};

using Message = std::variant<TextMessage, NumberMessage, QuitMessage>;

// 处理消息的 visitor
struct MessageHandler
{
    void operator()(const TextMessage& msg) const
    {
        std::cout << "Text: " << msg.text << "\n";
    }
    void operator()(const NumberMessage& msg) const
    {
        std::cout << "Number: " << msg.number << "\n";
    }
    void operator()(const QuitMessage&) const
    {
        std::cout << "Quit!\n";
    }
};

int main()
{
    Message msg;

    msg = TextMessage{"Hello World"};
    std::visit(MessageHandler{}, msg);

    msg = NumberMessage{42};
    std::visit(MessageHandler{}, msg);

    msg = QuitMessage{};
    std::visit(MessageHandler{}, msg);

    return 0;
}

运行结果

Text: Hello World
Number: 42
Quit!

运行结果

见上方每个示例的"运行结果"。

示例中的关键语法解释

示例讲了什么新出现的语法为什么这样写注意事项
示例 1基本赋值和 getstd::variant<int, double, string>std::get<T>(v)variant 类型安全,赋值时自动切换类型get<T> 类型不对会抛异常
示例 2get_if 安全访问std::get_if<T>(&v)返回指针,类型不对返回 nullptr比 get 更安全,推荐使用
示例 3visit 模式std::visit(lambda, v)visit 强制覆盖所有类型,是 variant 的最佳访问方式泛型 lambda + visit 是最简洁的组合
示例 4消息分发模式struct visitor + variant用 variant + visitor 实现类型安全的消息处理visitor 必须为每种类型都提供 operator()

variant 适合"有限几种类型之一"

variant 不是为了替代所有继承和多态。它最适合这种情况:类型种类有限,而且你希望编译器提醒你把每种情况都处理掉。

场景推荐
消息只有 Text / Number / Quit 三类std::variant
状态只有 Idle / Running / Error 几类std::variant
解析结果可能是 int / double / stringstd::variant
类型种类很多且需要运行时扩展插件继承 + 虚函数
所有对象共享一套接口多态接口更自然

示例 5:用 variant 表示状态机

#include <iostream>
#include <string>
#include <type_traits>
#include <variant>

struct Idle {};
struct Running
{
    int task_id;
};
struct Error
{
    std::string message;
};

// variant 表示一个变量可以在多个候选类型中保存其中一种。
using State = std::variant<Idle, Running, Error>;

void print_state(const State& state)
{
    // visit 会根据 variant 当前保存的类型调用对应处理逻辑。
    std::visit([](const auto& s) {
        using T = std::decay_t<decltype(s)>;

        if constexpr (std::is_same_v<T, Idle>)
        {
            std::cout << "state: idle\n";
        }
        else if constexpr (std::is_same_v<T, Running>)
        {
            std::cout << "state: running task " << s.task_id << "\n";
        }
        else if constexpr (std::is_same_v<T, Error>)
        {
            std::cout << "state: error " << s.message << "\n";
        }
    }, state);
}

int main()
{
    // 程序从 main 函数开始执行,下面的语句会按顺序运行。
    State state = Idle{};
    print_state(state);

    state = Running{42};
    print_state(state);

    state = Error{"motor timeout"};
    print_state(state);

    return 0;
}

运行结果

state: idle
state: running task 42
state: error motor timeout

这里的状态永远只能是三种之一。相比用 int state_code 加一堆额外字段,variant 能把每种状态需要的数据放在对应类型里,减少“错误状态却还读 running 字段”这类问题。

常见错误

错误 1:get 用错类型抛异常

std::variant<int, double> v = 42;
std::cout << std::get<double>(v);  // ❌ 抛出 std::bad_variant_access!

正确做法:先用 std::holds_alternative<double>(v) 检查,或用 std::get_if

错误 2:variant 中没有默认类型时默认构造

std::variant<int, std::string> v;  // 默认构造第一个类型的默认值(int = 0)

这种情况是合法的,但如果第一种类型没有默认构造函数,则编译失败。

错误 3:visit 的 visitor 没有覆盖所有类型

struct Visitor {
    void operator()(int) {}
    // 缺少 double 和 string 的 operator()
};
std::variant<int, double, std::string> v;
std::visit(Visitor{}, v);  // ❌ 编译错误!

正确做法:visit 的 visitor 必须为 variant 中所有类型提供 operator(),或者用泛型 lambda。

使用建议

  1. 替代 union:variant 类型安全,知道当前存的是什么。
  2. std::visit + 泛型 lambda 是最简洁的访问方式
  3. 需要"知道当前是哪种类型"时用 std::get_if:返回指针,安全高效。
  4. 用 variant + visit 实现消息/事件分发:模式匹配的雏形。
  5. variant 的大小是所有类型中最大的 + 索引字段:不要存太多大类型。
  6. 类型种类有限时用 variant 更清晰:如果类型需要随插件扩展,继承和虚函数通常更合适。

小结

  • std::variant<T1, T2, ...> 是类型安全的联合体。
  • std::get<T>(v) 直接获取(不安全),std::get_if<T>(&v) 返回指针(安全)。
  • std::visit(visitor, v) 是最推荐的方式,强制覆盖所有类型。
  • 适用于消息分发、可选配置、状态机等场景。
音乐页