Skip to content

Latest commit

 

History

History
389 lines (280 loc) · 8.27 KB

041-cpp17-lib-optional.md

File metadata and controls

389 lines (280 loc) · 8.27 KB

optional : 値を保有しているか、していないクラス

使い方

ヘッダーファイル<optional>で定義されているoptional<T>は、T型の値を保有しているか、保有していないライブラリだ。

条件次第で値が用意できない場合が存在する。例えば割り算の結果の値を返す関数を考える。

int divide( int a, int b )
{
    if ( b == 0 )
    {
        // エラー処理
    }
    else
        return a / b ;
}

ゼロで除算はできないので、bの値が0の場合、この関数は値を用意することができない。問題は、int型のすべての値は通常の除算結果として使われるので、エラーであることを示す特別な値を返すこともできない。

このような場合にエラーや値を通知する方法として、過去に様々な方法が考案された。例えば、ポインターやリファレンスを実引数として受け取る方法、グローバル変数を使う方法、例外だ。

optionalはこのような値が用意できない場合に使える共通の方法を提供する。

std::optional<int> divide( int a, int b )
{
    if ( b == 0 )
        return {} ;
    else
        return { a / b } ;
}

int main()
{
    auto result = divide( 10, 2 ) ;
    // 値の取得
    auto value = result.value() ;

    // ゼロ除算
    auto fail = divide( 10, 0 ) ;

    // false、値を保持していない
    bool has_value = fail.has_value() ;

    // throw bad_optional_access
    auto get_value_anyway = fail.value() ;
}

optionalのテンプレート実引数

optional<T>はT型の値を保持するか、もしくは保持しない状態を取る。

int main()
{
    // int型の値を保持するかしないoptional
    using a = std::optional<int> ;
    // double型の値を保持するかしないoptional
    using b = std::optional<double> ;
}

optionalの構築

optionalをデフォルト構築すると、値を保持しないoptionalになる。

int main()
{
    // 値を保持しない
    std::optional<int> a ;
}

コンストラクターの実引数にstd::nulloptを渡すと、値を保持しないoptionalになる。

int main()
{
    // 値を保持しない
    std::optional<int> a( std::nullopt ) ;
}

optional<T>のコンストラクターの実引数にT型に変換できる型を渡すと、T型の値に型変換して保持する。

int main()
{
    // int型の値42を保持する
    std::optional<int> a(42) ;

    // double型の値1.0を保持する
    std::optional<double> b( 1.0 ) ;

    // intからdoubleへの型変換が行われる
    // int型の値1を保持する
    std::optional<int> c ( 1.0 ) ;
}

T型からU型に型変換できるとき、optional<T>のコンストラクターにoptional<U>を渡すとUからTに型変換されてT型の値を保持するoptionalになる。

int main()
{
    // int型の値42を保持する
    std::optional<int> a( 42 ) ;

    // long型の値42を保持する
    std::optional<long> b ( a ) ;
}

optionalのコンストラクターの第一引数にstd::in_place_type<T>を渡すと、後続の引数を使ってT型のオブジェクトがemplace構築される。

struct X
{
    X( int, int ) { }
} ;

int main()
{
    // X(1, 2)
    std::optional<X> o( std::in_place_type<X>, 1, 2 ) ;
}

optionalの代入

通常のプログラマーの期待通りの挙動をする。std::nulloptを代入すると値を保持しないoptionalになる。

optionalの破棄

optionalが破棄されるとき、保持している値があれば、適切に破棄される。

struct X
{
    ~X() { }
} ;

int main()
{
    {
        // 値を保持する
        std::optional<X> o ( X{} ) ;
        // Xのデストラクターが呼ばれる。
    }

    {
        // 値を保持しない
        std::optional<X> o ;
        // Xのデストラクターは呼ばれない。
    }   
}

swap

optionalはswapに対応している。

int main()
{
    std::optional<int> a(1), b(2) ;

    a.swap(b) ;
}

has_value : 値を保持しているかどうか確認する

constexpr bool has_value() const noexcept;

has_valueメンバー関数はoptionalが値を保持している場合、trueを返す。

int main()
{
    std::optional<int> a ;
    // false
    bool b1 = a.has_value() ;

    std::optional<int> b(42) ;
    // true
    bool b2 = b.has_value() ;
}

value : 保持している値を取得

constexpr const T& value() const&;
constexpr T& value() &;
constexpr T&& value() &&;
constexpr const T&& value() const&&;

valueメンバー関数はoptionalが値を保持している場合、値へのリファレンスを返す。値を保持していない場合、std::bad_optional_accessがthrowされる。

int main()
{
    std::optional<int> a(42) ;

    // OK
    int x = a.value () ;

    try {
        std::optional<int> b ;
        int y = b.value() ;
    } catch( std::bad_optional_access e )
    {
        // 値を保持していなかった
    }
}

value_or : 値もしくはデフォルト値を返す

template <class U> constexpr T value_or(U&& v) const&;
template <class U> constexpr T value_or(U&& v) &&;

value_or(v)メンバー関数は、optionalが値を保持している場合はその値を、保持していない場合はvを返す。

int main()
{
    std::optional<int> a( 42 ) ;

    // 42
    int x = a.value_or(0) ;

    std::optional<int> b ;

    // 0
    int x = b.value_or(0) ;
}

reset : 保持している値を破棄する

resetメンバー関数を呼び出すと、保持している値がある場合破棄する。resetメンバー関数を呼び出した後のoptionalは値を保持しない状態になる。

int main()
{
    std::optional<int> a( 42 ) ;

    // true
    bool b1 = a.has_value() ;

    a.reset() ;

    // false
    bool b2 = a.has_value() ;
}

optional同士の比較

optional<T>を比較するためには、T型のオブジェクト同士が比較できる必要がある。

同一性の比較

値を保持しない二つのoptionalは等しい。片方のみが値を保持しているoptionalは等しくない。両方とも値を保持しているoptionalは値による比較になる。

int main()
{
    std::optional<int> a, b ;

    // true
    // どちらも値を保持しないoptional
    bool b1 = a == b ;

    a = 0 ;

    // false
    // aのみ値を保持
    bool b2 = a == b ;

    b = 1 ;

    // false
    // どちらも値を保持。値による比較
    bool b3 = a == b ;
}

大小比較

optional同士の大小比較は、a < bの場合

  1. bが値を保持していなければfalse
  2. それ以外の場合で、aが値を保持していなければtrue
  3. それ以外の場合、aとbの保持している値同士の比較

となる。

int main()
{
    std::optional<int> a, b ;

    // false
    // bが値なし
    bool b1 = a < b ;

    b = 0 ;

    // true
    // bは値ありでaが値なし
    bool b2 = a < b ;

    a = 1 ;

    // false
    // どちらとも値があるので値同士の比較
    // 1 < 0はfalse
    bool b3 = a < b ;
}

optionalとstd::nulloptとの比較

optionalとstd::nulloptとの比較は、std::nulloptが値を持っていないoptionalとして扱われる。

optional<T>とTの比較

optional<T>とT型の比較をする場合、optionalは値を保持していなければならない。

make_optional<T> : optional<T>を返す

template <class T>
constexpr optional<decay_t<T>> make_optional(T&& v);

make_optional<T>(T t)はoptional<T>(t)を返す。

int main()
{
    // std::optional<int>、値は0
    auto o1 = std::make_optional( 0 ) ;

    // std::optional<double>、値は0.0
    auto o2 = std::make_optional( 0.0 ) ;
}

make_optional<T, Args ... > : optional<T>をin_place_type構築して返す

make_optionalの第一引数がT型ではない場合、in_place_type構築するオーバーロード関数が選ばれる。

struct X
{
    X( int, int ) { }
} ;

int main()
{
    // std::optional<X>( std::in_place_type<X>, 1, 2 )
    auto o = std::make_optional<X>( 1, 2 ) ;
}