标准库标头 <optional> (C++17)学习之optional

2024-09-04 01:28

本文主要是介绍标准库标头 <optional> (C++17)学习之optional,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

类模板 std::optional 管理一个可选 的所含值,即既可以存在也可以不存在的值。

一种常见的 optional 使用情况是作为可能失败的函数的返回值。与如 std::pair<T, bool> 等其他手段相比,optional 可以很好地处理构造开销高昂的对象,并更加可读,因为它明确表达了意图。

optional<T> 的任何实例在任意给定时间点要么含值,要么不含值

如果一个 optional<T> 含值,那么保证该值作为 optional 对象所用空间的一部分分配,即不会发生动态内存分配。因此,optional 对象模拟的是对象而非指针,尽管定义了 operator*() 和 operator->() 运算符。

当一个 optional<T> 类型的对象被按语境转换到 bool 时,若对象含值 则转换返回 true,若它不含值" 则返回 false。

optional 对象在下列条件下含值

  • 对象被以 T 类型的值或另一含值 的 optional 初始化/赋值。

对象在下列条件下不含值

  • 对象被默认初始化。
  • 对象被以 std::nullopt_t 类型的值或不含值 的 optional 对象初始化/赋值。
  • 调用了成员函数 reset()。

不存在可选的引用、函数、数组或 cv void:如果以这些类型实例化 optional,那么程序非良构。另外,如果以(可有 cv 限定的)标签类型 std::nullopt_t 或 std::in_place_t 实例化 optional,那么程序非良构。

成员函数

(构造函数)

构造 optional 对象
(公开成员函数)

(析构函数)

销毁容纳的值(如果存在)
(公开成员函数)

operator=

对内容赋值
(公开成员函数)
观察器

operator->operator*

访问所含值
(公开成员函数)

operator boolhas_value

检查对象是否含值
(公开成员函数)

value

返回所含值
(公开成员函数)

value_or

在所含值可用时返回它,否则返回另一个值
(公开成员函数)
修改器

swap

交换内容
(公开成员函数)

reset

销毁任何所含值
(公开成员函数)

emplace

原位构造所含值
(公开成员函数)

非成员函数

make_optional

(C++17)

创建一个 optional 对象
(函数模板)

示例代码:

#include <iostream>
#include <optional>
#include <string>
#include <iomanip>
#include <cstdlib>
#include <vector>#pragma warning(disable:4996)// optional 可用作可能失败的工厂的返回类型
std::optional<std::string> create(bool b)
{if (b)return "Godzilla";return {};
}// 能用 std::nullopt 创建任何(空的)std::optional
auto create2(bool b)
{return b ? std::optional<std::string>{"Godzilla"} : std::nullopt;
}std::optional<const char*> maybe_getenv(const char* n)
{if (const char* x = std::getenv(n))return x;elsereturn {};
}struct A
{std::string s;A(std::string str) : s(std::move(str)) { std::cout << " 已构造\n"; }~A() { std::cout << " 已析构\n"; }A(const A& o) : s(o.s) { std::cout << " 被复制构造\n"; }A(A&& o) : s(std::move(o.s)) { std::cout << " 被移动构造\n"; }A& operator=(const A& other){s = other.s;std::cout << " 被复制赋值\n";return *this;}A& operator=(A&& other){s = std::move(other.s);std::cout << " 被移动赋值\n";return *this;}
};struct B
{std::string s;B(std::string str) : s(std::move(str)), id{ n++ } { note("+ 构造"); }~B() { note("~ 析构"); }B(const B& o) : s(o.s), id{ n++ } { note("+ 复制构造"); }B(B&& o) : s(std::move(o.s)), id{ n++ } { note("+ 移动构造"); }B& operator=(const B& other){s = other.s;note("= 复制赋值");return *this;}B& operator=(B&& other){s = std::move(other.s);note("= 移动赋值");return *this;}inline static int n{};int id{};void note(std::string s) { std::cout << "  " << s << " #" << id << '\n'; }
};int main()
{std::cout << "create(false) 返回 "<< create(false).value_or("empty") << '\n';// 返回 optional 的工厂函数可用作 while 和 if 的条件if (auto str = create2(true))std::cout << "create2(true) 返回 " << *str << '\n';//operator= example std::optional<const char*> s1 = "abcefg", s2; // 构造函数s2 = s1; // 赋值s1 = "hijklm"; // 衰变赋值(U = char[4], T = const char*)std::cout << *s2 << ' ' << *s1 << '\n';//std::optional<T>::operator->, std::optional<T>::operator*  exampleusing namespace std::string_literals;std::optional<int> opt1 = 1;std::cout << "opt1: " << *opt1 << '\n';*opt1 = 2;std::cout << "opt1: " << *opt1 << '\n';std::optional<std::string> opt2 = "abc"s;std::cout << "opt2: " << std::quoted(*opt2) << ", size: " << opt2->size() << '\n';// 你能通过在到 optional 的右值上调用 operator* “取”其所含值auto taken = *std::move(opt2);std::cout << "taken: " << std::quoted(taken) << "\n""opt2: " << std::quoted(*opt2) << ", size: " << opt2->size() << '\n';//std::optional<T>::operator bool, std::optional<T>::has_value examplestd::cout << std::boolalpha;std::optional<int> opt;std::cout << opt.has_value() << '\n';opt = 43;if (opt)std::cout << "设置值为 " << opt.value() << '\n';elsestd::cout << "未设置值\n";opt.reset();if (opt.has_value())std::cout << "值仍被设为 " << opt.value() << '\n';elsestd::cout << "不再设置值\n";//std::optional<T>::value examplestd::optional<int> opt3 = {};try{[[maybe_unused]] int n = opt3.value();}catch (const std::bad_optional_access& e){std::cout << e.what() << '\n';}try{opt3.value() = 42;}catch (const std::bad_optional_access& e){std::cout << e.what() << '\n';}opt3 = 43;std::cout << *opt3 << '\n';opt3.value() = 44;std::cout << opt3.value() << '\n';//std::optional<T>::value_or  examplestd::cout << maybe_getenv("SHELL").value_or("(none)") << '\n';std::cout << maybe_getenv("MYPWD").value_or("(none)") << '\n';//std::optional<T>::swap examplestd::optional<std::string> opt4("First example text");std::optional<std::string> opt5("2nd text");enum Swap { Before, After };auto print_opts = [&](Swap e) {std::cout << (e == Before ? "交换前:\n" : "交换后:\n");std::cout << "opt1 含有 '" << opt4.value_or("") << "'\n";std::cout << "opt2 含有 '" << opt5.value_or("") << "'\n";std::cout << (e == Before ? "---SWAP---\n" : "\n");};print_opts(Before);opt4.swap(opt5);print_opts(After);// 在仅一者含值时交换opt4 = "Lorem ipsum dolor sit amet, consectetur tincidunt.";opt5.reset();print_opts(Before);opt4.swap(opt5);print_opts(After);//std::optional<T>::reset examplestd::cout << "创建空 optional:\n";std::optional<A> opt6;std::cout << "创建并赋值:\n";opt6 = A("Lorem ipsum dolor sit amet, consectetur adipiscing elit nec.");std::cout << "重置 optional:\n";opt6.reset();std::cout << "A示例结束\n";//emaplace examplestd::optional<B> opt7;std::cout << "赋值:\n";opt7 = B("Lorem ipsum dolor sit amet, consectetur adipiscing elit nec.");std::cout << "放置:\n";// 由于 opt 含值,这亦将销毁该值opt7.emplace("Lorem ipsum dolor sit amet, consectetur efficitur. ");std::cout << "B示例结束\n";//std::make_optional exampleauto op8 = std::make_optional<std::vector<char>>({ 'a','b','c' });std::cout << "op1: ";for (char c : op8.value())std::cout << c << ',';auto op9 = std::make_optional<std::vector<int>>(5, 2);std::cout << "\nop2: ";for (int i : *op9)std::cout << i << ',';std::string str{ "hello world" };auto op10 = std::make_optional<std::string>(std::move(str));std::cout << "\nop3: " << quoted(op10.value_or("empty value")) << '\n';std::cout << "str: " << std::quoted(str) << '\n';}

运行结果:

参考:

std::optional - cppreference.com

这篇关于标准库标头 <optional> (C++17)学习之optional的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1134607

相关文章

Java学习手册之Filter和Listener使用方法

《Java学习手册之Filter和Listener使用方法》:本文主要介绍Java学习手册之Filter和Listener使用方法的相关资料,Filter是一种拦截器,可以在请求到达Servl... 目录一、Filter(过滤器)1. Filter 的工作原理2. Filter 的配置与使用二、Listen

C++如何通过Qt反射机制实现数据类序列化

《C++如何通过Qt反射机制实现数据类序列化》在C++工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作,所以本文就来聊聊C++如何通过Qt反射机制实现数据类序列化吧... 目录设计预期设计思路代码实现使用方法在 C++ 工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作。由于数据类

Linux下如何使用C++获取硬件信息

《Linux下如何使用C++获取硬件信息》这篇文章主要为大家详细介绍了如何使用C++实现获取CPU,主板,磁盘,BIOS信息等硬件信息,文中的示例代码讲解详细,感兴趣的小伙伴可以了解下... 目录方法获取CPU信息:读取"/proc/cpuinfo"文件获取磁盘信息:读取"/proc/diskstats"文

Java Optional的使用技巧与最佳实践

《JavaOptional的使用技巧与最佳实践》在Java中,Optional是用于优雅处理null的容器类,其核心目标是显式提醒开发者处理空值场景,避免NullPointerExce... 目录一、Optional 的核心用途二、使用技巧与最佳实践三、常见误区与反模式四、替代方案与扩展五、总结在 Java

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

C++中初始化二维数组的几种常见方法

《C++中初始化二维数组的几种常见方法》本文详细介绍了在C++中初始化二维数组的不同方式,包括静态初始化、循环、全部为零、部分初始化、std::array和std::vector,以及std::vec... 目录1. 静态初始化2. 使用循环初始化3. 全部初始化为零4. 部分初始化5. 使用 std::a

C++ vector的常见用法超详细讲解

《C++vector的常见用法超详细讲解》:本文主要介绍C++vector的常见用法,包括C++中vector容器的定义、初始化方法、访问元素、常用函数及其时间复杂度,通过代码介绍的非常详细,... 目录1、vector的定义2、vector常用初始化方法1、使编程用花括号直接赋值2、使用圆括号赋值3、ve

如何高效移除C++关联容器中的元素

《如何高效移除C++关联容器中的元素》关联容器和顺序容器有着很大不同,关联容器中的元素是按照关键字来保存和访问的,而顺序容器中的元素是按它们在容器中的位置来顺序保存和访问的,本文介绍了如何高效移除C+... 目录一、简介二、移除给定位置的元素三、移除与特定键值等价的元素四、移除满足特android定条件的元

Python获取C++中返回的char*字段的两种思路

《Python获取C++中返回的char*字段的两种思路》有时候需要获取C++函数中返回来的不定长的char*字符串,本文小编为大家找到了两种解决问题的思路,感兴趣的小伙伴可以跟随小编一起学习一下... 有时候需要获取C++函数中返回来的不定长的char*字符串,目前我找到两种解决问题的思路,具体实现如下:

Java Optional避免空指针异常的实现

《JavaOptional避免空指针异常的实现》空指针异常一直是困扰开发者的常见问题之一,本文主要介绍了JavaOptional避免空指针异常的实现,帮助开发者编写更健壮、可读性更高的代码,减少因... 目录一、Optional 概述二、Optional 的创建三、Optional 的常用方法四、Optio