C++(25): C++17探索:std::optional

1. 概述

std::optional是C++17标准引入的一个新的标准库组件。这是一个轻量级的容器,旨在以类型安全的方式包装可能为空的值。std::optional为处理可能缺失的数据提供了一种优雅而简洁的方法,这在以往通常需要使用专门的"存在"标志或复杂的错误处理机制。

简单点讲,std::optional在使用上并不复杂,主要用来替代那些需要返回NULL,或以返回-1表示错误等的场合。这样做最直观的一个改进就是易读性增强了。

2. 为什么需要std::optional?

在C++中,函数通常返回一个值或者在失败时返回一个特殊值(如NULL或-1)。这种方法存在几个问题:

(1)类型不安全:返回特殊值通常意味着函数返回的类型与实际返回值的类型不一致,这可能导致类型安全问题。

(2)语义不明确:使用特殊值来表示错误状态可能会与合法的返回值冲突。

(3)错误处理复杂:调用者需要检查返回的特殊值,并进行额外的错误处理。

std::optional提供了一种更好的解决方案,它允许函数明确地表达它们可能不返回有效值的情况。

3. std::optional的基本概念

std::optional<T>是一个模板类,其中T是被包装的值的类型。它包含以下两个主要状态:

无值状态:表示不包含有效的T类型对象。

有值状态:包含一个有效的T类型对象。

4. 使用std::optional

(1)构造

cpp 复制代码
std::optional<int> opt;         // 默认构造,无值状态

std::optional<int> opt(10);     // 有值状态,包含值10

(2)有无值判断has_value()

返回一个布尔类型,如果当前容器是有值状态,返回true,否则返回false。

使用方法如下:

cpp 复制代码
if (opt.has_value()) {

    // opt包含一个值

}

需要注意的是,当我们想要访问std::optional类型的值时,需要先判断是否有值,否则会出现未定义行为。

(3)使用operator*或value()访问存储的值

首先我们需要判断是否有值,然后再进行访问。

或者添加异常捕获单元。

cpp 复制代码
#include <optional>

#include <iostream>


int main(int argc, char* argv[])

{

    std::optional<int> opt(10);

    if (opt) {

        std::cout << "Value: " << *opt << std::endl; ///< 输出 10

    }

    return 0;

}

cpp 复制代码
int main(int argc, char* argv[])

{

    std::optional<int> opt;


    try {

        int value = opt.value(); ///< 这里会抛出异常

    } catch (const std::bad_optional_access& e) {

        std::cerr << "Exception: " << e.what() << std::endl;

    }

    return 0;

}

(4)value_or(T default):返回存储的值,如果std::optional处于无值状态,则返回提供的默认值。

cpp 复制代码
int value = opt.value_or(-1);  ///< 返回10,如果opt无值则返回-1

(5)reset():将std::optional重置为无值状态。

cpp 复制代码
opt.reset();  ///< opt变为无值状态

(6)emplace():在原地构造一个新值,如果std::optional已经有值,则先销毁旧值。

cpp 复制代码
opt.emplace(20);  // opt现在包含20
相关推荐
张健11564096485 小时前
使用信号量限制并发数量
开发语言·c++
jc06205 小时前
6.1云原生之Docker
c++·docker·云原生
叶子野格8 小时前
《C语言学习:指针》12
c语言·开发语言·c++·学习·visual studio
Fuyo_11199 小时前
C++ 内存管理
c++·笔记
澈2079 小时前
C++面向对象:类与对象核心解析
c++·算法
6Hzlia10 小时前
【Hot 100 刷题计划】 LeetCode 141. 环形链表 | C++ 哈希表直觉解法
c++·leetcode·链表
handler0110 小时前
Linux 进程探索:从 PCB 管理到 fork() 的写时拷贝
linux·c语言·c++·笔记·学习
众少成多积小致巨11 小时前
GNU Make 核心指南
android·c++
谭欣辰11 小时前
详细讲解 C++ 状压 DP
开发语言·c++·动态规划
William_wL_12 小时前
【C++】stack和queue的使用和实现(附加deque的简单介绍)
开发语言·c++