ヘッダーファイル<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<T>はT型の値を保持するか、もしくは保持しない状態を取る。
int main()
{
// int型の値を保持するかしないoptional
using a = std::optional<int> ;
// double型の値を保持するかしないoptional
using b = std::optional<double> ;
}
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 ) ;
}
通常のプログラマーの期待通りの挙動をする。std::nulloptを代入すると値を保持しないoptionalになる。
optionalが破棄されるとき、保持している値があれば、適切に破棄される。
struct X
{
~X() { }
} ;
int main()
{
{
// 値を保持する
std::optional<X> o ( X{} ) ;
// Xのデストラクターが呼ばれる。
}
{
// 値を保持しない
std::optional<X> o ;
// Xのデストラクターは呼ばれない。
}
}
optionalはswapに対応している。
int main()
{
std::optional<int> a(1), b(2) ;
a.swap(b) ;
}
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() ;
}
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 )
{
// 値を保持していなかった
}
}
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メンバー関数を呼び出した後のoptionalは値を保持しない状態になる。
int main()
{
std::optional<int> a( 42 ) ;
// true
bool b1 = a.has_value() ;
a.reset() ;
// false
bool b2 = a.has_value() ;
}
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の場合
- bが値を保持していなければfalse
- それ以外の場合で、aが値を保持していなければtrue
- それ以外の場合、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との比較は、std::nulloptが値を持っていないoptionalとして扱われる。
optional<T>とT型の比較をする場合、optionalは値を保持していなければならない。
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型ではない場合、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 ) ;
}