C++17使用std::optional表示可能存在的值

目录
  • 前言
  • 返回一个bool值
  • 使用 std::optional 改写
  • 总结

前言

平时写代码会遇到一种传递参数特殊值标记特殊流程,或者函数返回值存在魔法数的情况,很需要一种标记参数或返回值状态的结构,那么在 C++17 标准下提供了 std::optional 这个模板类,可以表示一个值不存在的状态,一起来看看用法吧。

返回一个bool值

以下例子纯属虚构,只为说明问题,无实际意义

bool getBoolVal(int a, int b)
{
    int* n = new int;
    if (!n)
        return false;

    *n = 1;

    if (a + *n > b)
        return true;
    else
        return false;
}

int main()
{
    if (getBoolVal(10, 9))
        std::cout << 1 << std::endl;
    else
        std::cout << 1 << std::endl;

    return 0;
}

这个例子中的函数 getBoolVal 本意是想返回一个 bool 类型的判断结果,但是函数中有一些异常情况时,比如申请内存异常时,也会返回一个bool值,这是与原判断结果语义不同的,所以需要单独返回这种情况,如果也放到同一个返回值中会导致含义模糊,这时可以考虑使用引用变量参数来返回实际比较结果。

bool getBoolVal(int a, int b, bool& ret)
{
    int* n = new int;
    if (!n)
        return false;

    *n = 1;

    if (a + *n > b)
        ret = true;
    else
        ret = false;

    return true;
}

int main()
{
    bool ret = false;
    if (getBoolVal(10, 9, ret))
        std::cout << "error" << std::endl;
    else
    {
        if (ret)
            std::cout << 1 << std::endl;
        else
            std::cout << 0 << std::endl;
    }

    return 0;
}

这个引用参数 ret 使用起来有点不方便,那把两个值都返回怎么样,虽然C++不允许有多个返回值,但可以把它们包装成 std::pair 或者 std::tuple 来返回,再来改写一下:

std::pair<bool, bool> getBoolVal3(int a, int b)
{
    int* n = new int;
    if (!n)
        return {false, false};

    *n = 1;

    if (a + *n > b)
        return {true, true};
    else
        return {true, false};
}

int main()
{
    auto [err, ret] = getBoolVal(10, 9);
    if (err)
        std::cout << "error" << std::endl;
    else
    {
        if (ret)
            std::cout << 1 << std::endl;
        else
            std::cout << 0 << std::endl;
    }

    return 0;
}

这种方法把实际的返回值,搭配一个表示状态的 bool 变量,组成 std::pair 进行返回,基本上得到而来语义明确的目的,但是看起来还是不太优雅,而 std::optional 可以帮助我们实现类似的需求,并且代码看起来能更简洁一点。

使用 std::optional 改写

std::optional 本身是一个模板类:会有一个 std::nullopt

template <class T>
class optional;

它内部有两种状态,要么有一个T类型的值,要么用 std::nullopt 表示没有值,查看一个 std::optional 对象是否有值,可以用 has_value() 进行判断,当一个 std::optional 有值时,可以通过用指针的方式(*号和->号)来使用它,或者用 value()函数取它的值,下面我们用它来改写一下之前的实现:

std::optional<bool> getBoolVal4(int a, int b)
{
    int* n = new int;
    if (!n)
        return std::nullopt;

    *n = 1;

    if (a + *n > b)
        return true;
    else
        return false;
}

int main()
{
    std::optional<bool> ret = getBoolVal(10, 9);
    if (ret.has_value())
        std::cout << "error" << std::endl;
    else
    {
        if (ret.value())
            std::cout << 1 << std::endl;
        else
            std::cout << 0 << std::endl;
    }

    return 0;
}

使用了 std::optional 之后就把 bool 类型之前的两态变成了三态,很多类似的逻辑也被封装成了函数,使用它之后代码更清晰了,从此可以告别一些烦人的魔法数了,一些函数参数也可以使用 std::optional 来包装,用法类似,在此就不展开说了。

总结

  • std::optional 是一个模板类,可以表示一个可能存在的值
  • std::optional 的内部有两种状态,要么表示一个T类型的值,要么用 std::nullopt 表示没有值
  • 可以用 has_value() 判断一个 std::optional 是否有值,然后用 value() 函数取它表示的值

到此这篇关于C++17使用std::optional表示可能存在的值的文章就介绍到这了,更多相关C++17 std::optional内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • C++17中的std::optional的具体使用

    直入主题 本篇之中,仅仅述及 std::optional ,其它和 variant 相关的话题以后再说吧. std::optional 也划入 variant 类别中,其实它还是谈不上可称为变体类型的,但新版本中的三大件(optional,any and variant)也可以归一类无妨. C++17 之前 在 C 时代以及早期 C++ 时代,语法层面支持的 nullable 类型可以采用指针方式: T* ,如果指针为 NULL (C++11 之后则使用 nullptr ) 就表示无值状态(em

  • C++17使用std::optional表示可能存在的值

    目录 前言 返回一个bool值 使用 std::optional 改写 总结 前言 平时写代码会遇到一种传递参数特殊值标记特殊流程,或者函数返回值存在魔法数的情况,很需要一种标记参数或返回值状态的结构,那么在 C++17 标准下提供了 std::optional 这个模板类,可以表示一个值不存在的状态,一起来看看用法吧. 返回一个bool值 以下例子纯属虚构,只为说明问题,无实际意义 bool getBoolVal(int a, int b) {     int* n = new int;  

  • C++17 使用 std::string_view避免字符串拷贝优化程序性能

    C++中std::string是日常Coding中经常使用的一个类,使用起来非常方便,但是也存在一些弊端. 如下代码,参数传递的过程发生了内存分配(Memory Allocation)和内存拷贝. void fun(const std::string& s) { std::cout << s << std::endl; } const char* ch = "hello world"; // bad way, expensive if the strin

  • C++17之std::any的具体使用

    目录 1. 使用std::any 2. std::any类型和操作 2.1 std::any的类型 2.2 std::any操作 一般来说,c++是一种具有类型绑定和类型安全性的语言.值对象声明为具有特定类型,该类型定义哪些操作是可能的以及它们的行为方式.值对象不能改变它们的类型. std: any是一种值类型,它能够更改其类型,同时仍然具有类型安全性.也就是说,对象可以保存任意类型的值,但是它们知道当前保存的值是哪种类型.在声明此类型的对象时,不需要指定可能的类型. 诀窍在于,对象同时拥有包含

  • C++17之std::visit的具体使用

    目录 1. 使用对象函数方式访问 2. 使用泛型Lambdas访问 3. 使用重载的Lambdas来访问 它们必须明确地为每种可能的类型提供函数调用操作符.然后,使用相应的重载来处理当前的备选项类型. 1. 使用对象函数方式访问 例1: #include <iostream> #include <variant> #include <string> struct MyVisitor { void operator()(double d) const { std::cou

  • C++ 17标准正式发布! 更简单地编写和维护代码

    C++17 是继 C++14 之后,C++ 编程语言 ISO/IEC 标准的下一次修订的非正式名称.而就在昨日,ISO C++ 委员会正式发布了 C++ 17 标准,官方名称为 ISO/IEC 14882:2017. C++ 17 标准化图表 C ++ 17 主要特性 基于 C++ 11,C++ 17 旨在使 C++ 成为一个不那么臃肿复杂的编程语言,以简化该语言的日常使用,使开发者可以更简单地编写和维护代码. C++ 17 是对 C++ 语言的重大更新,引入了许多新的语言特性: UTF-8 字

  • C++ 17转发一个函数调用的完美实现

    前言 本文主要给大家介绍了关于C++17转发一个函数调用的相关内容,分享出来供大家参考学习,下面话不多说了,来一起看看详细的介绍吧. 方法如下 首先你灵光一闪: #define WARP_CALL(fun, ...) fun(__VA_ARGS__) 不我们并不喜欢宏,扩展性太差了 template<class R, class T1, class T2, class T3> R warp_call(R(*fun)(T1, T2, T3), T1 a, T2 b, T3 c) { return

  • Java Optional解决空指针异常总结(java 8 功能)

    1.概述 Java8的版本,新增了Optional和[Lambda]表达式,Optional主要用于作为返回类型(主要解决的问题是臭名昭著的空指针异常 (NullPointerException)),并将其与流(或返回可选的方法)相结合以构建连贯API. 但是,有些情况可以被认为是陷阱,因为它们会降低代码的质量,甚至导致意想不到的错误.总结以下26个例子,以避免这些陷阱. 2. 目 录 [第1项:决不将Null分配给可选变量] [第2项:调用Optional.get()之前,确保Optional

  • Swift中Optional值的链式调用学习笔记

    Swift中的Optional值有这样的特性,当对其进行可选拆包时,即使用?进行Optional类型值的取值时,如果Optional值不为nil,则会返回原始类型的数据值,如果为nil,则会返回nil.因此,当使用?对Optional拆包后进行方法.属性或者下标的调用时,如果有值,则会成功相应调用,如果没有值,则会调用失败,返回nil. 注意:使用!则会进行强制拆包,这时如果Optional值为nil,则会出现运行时错误,因此开发者在使用!进行强制拆包时,必须确认Optional类型值不为nil

  • C++11右值引用和std::move语句实例解析(推荐)

    右值引用(及其支持的Move语意和完美转发)是C++0x将要加入的最重大语言特性之一.从实践角度讲,它能够完美解决C++中长久以来为人所诟病的临时对象效率问题.从语言本身讲,它健全了C++中的引用类型在左值右值方面的缺陷.从库设计者的角度讲,它给库设计者又带来了一把利器.从库使用者的角度讲,不动一兵一卒便可以获得"免费的"效率提升- 下面用实例来深入探讨右值引用. 1.什么是左值,什么是右值,简单说左值可以赋值,右值不可以赋值.以下面代码为例,"A a = getA();&q

随机推荐