当前位置: 首页 > news >正文

C++17中std::optional的使用

      模版类std::optional管理一个可选的(optional)存储值(contained value),即可能存在也可能不存在的值。std::optional的一个常见用例是存储可能失败的函数的返回值。与其它方法相反(例如std::pair<T, bool>),std::optional可以很好地处理构造成本高昂的对象,并且更具可读性,因为意图被明确表达。
      在任何给定时间点的std::optional的任何实例要么存储值,要么不存储值。如果std::optional存储一个值,则保证该值作为std::optional对象占用空间的一部分进行分配,即不会发生动态内存分配。
      当std::optional<T>类型的对象根据上下文转换为bool时,如果该对象存储一个值,则转换返回true;如果不存储值,则转换返回false。
      和std::variant、std::any一样,std::optional对象有值语义,其拷贝操作会被实现为深拷贝。std::optional对象也支持move语义
      std::optional模拟了一个可以为空的任意类型的实例,它可以被用作成员、参数、返回值等。

      std::optional提供了一种标准化的方式来表示可选值(optional value),消除了对空指针的需要并增强了代码的可读性。std::optional是使用原始指针或其它机制来表示缺失值或可选值的更安全的替代方案。根据设计,std::optional强制用户在访问某个值之前显式检查该值是否存在,从而最大限度地降低空指针解引用错误的风险。

      std::make_optional:创建一个用参数初始化的std::optional对象。

      std::nullopt是std::nullopt_t类型的常量,用于指示具有未初始化(或空的)状态的std::optional类型,作为std::optional对象无值时候的"值".
      std::optional特别适合延迟初始化问题,它本身就是延迟初始化的一个实例。所存储的T可以在构造时初始化,或者稍后初始化,或者从不初始化。

int test_optional_init()
{std::optional<int> var;if (!var.has_value())std::cout << "var is not set\n"; // var is not setvar = 8; // error: *var = 8, 因为var采用默认初始化为空if (var.has_value())std::cout << "var is present: " << var.value() << "\n"; // var is present: 8if (var)std::cout << "var is present: " << *var << "\n"; // var is present: 8std::optional<int> var2{ std::nullopt }; // 将var2初始化置于"empty"状态if (!var2.has_value())std::cout << "var2 is not set\n"; // var2 is not setstd::optional<std::string> var3{ "China" };if (var3 > std::nullopt)std::cout << "var3 > std::nullopt\n"; // var3 > std::nulloptauto var4 = std::make_optional<std::vector<char*>>({ "China", "Beijing", "Haidian" });for (const auto& v : *var4)std::cout << " " << v; //  China Beijing Haidianstd::cout << "\n";// 内建类型可不需指明存储类型,deduction guidestd::optional var5{ 2 }; std::cout << "var5: " << var5.value() << "\n"; // var5: 2std::optional var6{ std::string{"China"} };std::cout << "var6: " << var6.value() << "\n"; // var6: Chinareturn 0;
}

      std::optional成员函数:
      1.访问存储值:
      (1).var.value():返回引用;如果没有值,会抛出std::bad_optional_access异常;
      (2).*var:如果没有值,会crash(or undefined behavior),返回存储值的引用;应该只用于已经确定含有值的场景
      (3).var.value_or:返回值,注意与value()的区别;如果没有值,则获取备选值;
      (4).var->:返回存储值的指针,可访问内部值的成员
      2.emplace:赋予一个新值(in-place);如果在调用之前已经存储一个值,则通过调用其析构函数来销毁所存储的值;
      3.reset:如果存储一个值,则销毁该值;否则,没有任何影响;
      4.has_value:返回std::optional对象是否存有值;
      5.operator bool():返回std::optional对象是否存有值;
      6.swap:交换两个std::optional对象的值;
      非成员函数:比较std::optional对象:operator==, !=, <, <=, >, >=, <=>。比较std::optional对象时,"empty"值即std::nullopt比任何有值的std::optional对象都要小
      std::bad_optional_access:是一个异常类,直接派生自std::exception,用于处理在访问std::optional对象时发生的错误,当尝试访问一个没有存储值的std::optional对象时,会抛出std::bad_optional_access异常。

int test_optional_member_functions()
{// 获取std::optional值: var.value(), *var, var.value_or, var->std::optional<int> var{ 8 };std::cout << "var: " << var.value() << ", " << *var << "\n"; // var: 8, 8*var = 6; std::cout << "var: " << *var << "\n"; // var: 6var = 8; std::cout << "var: " << *var << "\n"; // var: 8*var = 10; std::cout << "var: " << var.value_or(66) << "\n"; // var: 10var.reset(); std::cout << "var: " << var.value_or(66) << "\n"; // var: 66std::optional<std::string> var2{ "China" };std::cout << "var2: " << var2->data() << "\n"; // var2: Chinavar.reset(); // or: var = {};try {std::cout << "var: " << var.value() << "\n";//std::cout << "var: " << *var << "\n"; // 直接crash掉}catch (const std::bad_optional_access& e) {std::cout << "exception: " << e.what() << "\n"; // var: exception: Bad optional access}std::string str{ "Beijing" }; // 不要对常量变量使用std::movestd::optional<std::string> var3(std::move(str));std::cout << "var3: " << var3.value() << ", str: " << str << "\n"; // var3: Beijing, str:var3.emplace("Haidian");std::cout << "var3: " << var3.value() << "\n"; // var3: Haidianif (var3.has_value())std::cout << "var3 has value\n"; // var3 has valueif (var3)std::cout << "var3 has value\n"; // var3 has valuestd::cout << "var2: " << var2.value() << ", var3: " << var3.value() << "\n"; // var2: China, var3: Haidianvar2.swap(var3);std::cout << "var2: " << var2.value() << ", var3: " << var3.value() << "\n"; // var2: Haidian, var3: Chinaif (var2 != var3)std::cout << "var2 != var3\n"; // var2 != var3return 0;
}

      执行结果如下图所示:

      GitHub:https://github.com/fengbingchun/Messy_Test

http://www.lryc.cn/news/226564.html

相关文章:

  • c++求三个数的最小公倍数
  • Flink 基础 -- 尝试Flink
  • kubeadm部署k8s及高可用
  • GEE:将鼠标变成十字指针,点击获取影像值,显示值到UI中
  • SpringBoot 项目公共字段填充
  • 分布式搜索引擎ES
  • 2023年05月 Python(四级)真题解析#中国电子学会#全国青少年软件编程等级考试
  • @KafkaListener注解详解(一)| 常用参数详解
  • 蓝桥杯算法双周赛心得——被替换的身份证(分类讨论)
  • 实验一:人工智能之启发式搜索算法(含源码+实验报告)
  • C++学习 --类和对象之封装
  • 在Spring Boot中使用JTA实现对多数据源的事务管理
  • 介绍YOLO-NAS Pose:姿势估计的技术
  • 计算机毕业设计 基于SpringBoot的实训管理系统的设计与实现 Java实战项目 附源码+文档+视频讲解
  • Python开发运维:Python3.7使用QQ邮箱发送不同类型邮件
  • 二十三种设计模式全面解析-解密迭代器模式:探索遍历之道
  • kubernetes istio
  • 25期代码随想录算法训练营第十四天 | 二叉树 | 递归遍历、迭代遍历
  • 常用布局以及其优缺点
  • 海康工业相机如何提高相机帧率
  • Linux之IPC通信共享内存(一次拷贝)与消息队列、管道、信号量、socket(两次拷贝)总结(六十二)
  • 【多线程 - 01、概述】
  • SQL SELECT INTO 语句
  • 【刷题】(AtCoder Beginner Contest 328) C、D 补题
  • NI USRP软件无线设备的特点
  • 大数据毕业设计选题推荐-污水处理大数据平台-Hadoop-Spark-Hive
  • 最新获取支付宝cardIndex参数图文教程
  • Linux学习第二枪(yum,vim,g++/gcc,makefile的使用)
  • 自然语言处理(一):RNN
  • 超全总结!大模型算法面试指南(含答案)