0

0

c++23的std::monostate在std::variant中有什么用? (空状态表示)

尼克

尼克

发布时间:2026-01-12 13:08:02

|

630人浏览过

|

来源于php中文网

原创

std::monostate 是专为 std::variant 设计的零大小默认可构造占位类型,用于显式表示“未初始化”状态;它不携带数据、无可比较性,仅通过 std::holds_alternative 安全识别,且不增加 variant 内存开销。

c++23的std::monostate在std::variant中有什么用? (空状态表示)

std::monostate 是 std::variant 的合法空状态类型

std::variant 中,不能定义“不含任何值”的实例(即没有默认构造出“空”状态),除非你显式提供一个可默认构造的类型作为其第一个备选项。而 std::monostate 就是标准库为此专门设计的“零大小、无状态、仅用于占位”的类型——它不携带数据、不比较相等、不参与逻辑判断,唯一作用就是让 std::variant 合法地拥有一个默认构造状态。

常见错误现象:

std::variant v; // 编译失败!因为 int 和 std::string 都不是默认可构造的?错:int 是默认可构造的,但问题在于——std::variant 默认构造时会尝试默认构造其第一个类型(int),这没问题;但如果你写的是 std::variant,而 std::string 默认构造是 OK 的,所以也 OK。真正需要 monostate 的场景是:你想让 variant 明确表达“尚未赋值”或“无效状态”,且不希望误用某个业务类型(比如用 0 表示 int 的“空”)来模拟空值。

  • std::monostate 大小为 0,无成员,无比较操作符(operator== 等需用户自定义),只支持赋值和销毁
  • 它被设计为 std::variant 的“哑占位符”,语义上代表“此处无有效值”,而非业务意义上的某个状态
  • 若你把 std::monostate 放在 std::variant 列表首位(如 std::variant<:monostate int std::string>),则默认构造后其 index() 返回 0,且 std::holds_alternative<:monostate>(v)true

为什么不用 std::nullopt 或 void?

std::nullopt_t 不是可构造/可存储类型(它是字面量类型,不可实例化变量),不能作为 std::variant 的模板参数;void 更不行——根本不是对象类型。只有满足“可默认构造 + 可析构 + 可复制/移动”的类型才能进 std::variant,而 std::monostate 正是为此定制的最小合规类型。

  • std::monostate{} 是合法表达式,能绑定到 const std::monostate&,能存入 std::variant
  • std::nulloptstd::nullopt_t 类型的常量,但 std::nullopt_t 没有默认构造函数,无法出现在 variant 模板参数列表中
  • 试图写 std::variant 直接编译失败:error: ‘void’ is not a valid type for a variant alternative

实际使用中怎么判断和切换空状态?

关键不是“怎么创建”,而是“怎么安全识别和处理”。std::monostate 本身不提供语义,你需要靠 std::holds_alternativestd::visit 显式分支处理。

Noya
Noya

让线框图变成高保真设计。

下载

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

std::variant v;
// 默认构造 → 持有 std::monostate
assert(v.index() == 0);
assert(std::holds_alternative(v));

// 赋值后切换
v = 42;
assert(v.index() == 1);
assert(std::holds_alternative(v));

// 使用 visit 处理所有情况(含 monostate)
std::visit([](const auto& x) {
    using T = std::decay_t;
    if constexpr (std::is_same_v) {
        // 这里处理“空”逻辑:日志、跳过、报错等
        std::cout << "uninitialized\n";
    } else if constexpr (std::is_same_v) {
        std::cout << "int: " << x << "\n";
    } else if constexpr (std::is_same_v) {
        std::cout << "string: " << x << "\n";
    }
}, v);
  • 永远不要依赖 v.index() == 0 来判断空——如果以后把 std::monostate 移到第二位,逻辑就崩了;应始终用 std::holds_alternative<:monostate>(v)
  • 不要对 std::monostate 做任何解引用或转型操作:它没有 value()、没有 get() 成员,强行调用会编译失败
  • 性能上无开销:std::monostate 不增加 std::variant 的大小,也不影响访问速度

替代方案对比:optional> vs variant

有人会想:我直接套一层 std::optional<:variant std::string>> 不也能表示“空”吗?可以,但语义和成本不同。

  • std::optional 额外占用 1 字节(对齐后可能更多)存储 has_value 标志;std::variant<:monostate ...> 把“空”当作一种合法替代项,复用已有 index 字段,空间更紧凑
  • std::optional<:variant>> 是两层嵌套:先判 optional 是否有值,再判 variant 持有哪个类型;而单层 variant + monostate 只需一次 std::visitstd::holds_alternative
  • 但注意:std::monostate 不等于 “错误/异常状态”——它只是“未初始化”,不代表操作失败。若需区分“未设置”和“设置失败”,仍应引入额外类型(如 std::expected

最易被忽略的一点:monostate 不提供任何调试线索。一旦 variant 持有它,你完全不知道它为何为空——是初始化遗漏?路径未覆盖?还是故意预留?必须配合上下文或额外标记(比如封装成类,加状态字段)才能避免误用。

相关文章

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

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

下载

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

相关专题

更多
string转int
string转int

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

315

2023.08.02

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1463

2023.10.24

scripterror怎么解决
scripterror怎么解决

scripterror的解决办法有检查语法、文件路径、检查网络连接、浏览器兼容性、使用try-catch语句、使用开发者工具进行调试、更新浏览器和JavaScript库或寻求专业帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

187

2023.10.18

500error怎么解决
500error怎么解决

500error的解决办法有检查服务器日志、检查代码、检查服务器配置、更新软件版本、重新启动服务、调试代码和寻求帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

271

2023.10.25

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

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

520

2023.09.20

string转int
string转int

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

315

2023.08.02

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

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

534

2024.08.29

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

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

51

2025.08.29

Java 项目构建与依赖管理(Maven / Gradle)
Java 项目构建与依赖管理(Maven / Gradle)

本专题系统讲解 Java 项目构建与依赖管理的完整体系,重点覆盖 Maven 与 Gradle 的核心概念、项目生命周期、依赖冲突解决、多模块项目管理、构建加速与版本发布规范。通过真实项目结构示例,帮助学习者掌握 从零搭建、维护到发布 Java 工程的标准化流程,提升在实际团队开发中的工程能力与协作效率。

4

2026.01.12

热门下载

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

精品课程

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

共18课时 | 4.4万人学习

Sass 教程
Sass 教程

共14课时 | 0.8万人学习

Pandas 教程
Pandas 教程

共15课时 | 0.9万人学习

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

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