std::expected<T,E>::operator=
Primary template |
||
constexpr expected& operator=( const expected& other ); |
(1) | (since C++23) |
constexpr expected& operator=( expected&& other ) noexcept(/* see below */); |
(2) | (since C++23) |
template< class U = T > constexpr expected& operator=( U&& v ); |
(3) | (since C++23) |
template< class G > constexpr expected& operator=( const std::unexpected<G>& e ); |
(4) | (since C++23) |
template< class G > constexpr expected& operator=( std::unexpected<G>&& e ); |
(5) | (since C++23) |
void partial specialization |
||
constexpr expected& operator=( const expected& other ); |
(6) | (since C++23) |
constexpr expected& operator=( expected&& other ) noexcept(/* see below */); |
(7) | (since C++23) |
template< class G > constexpr expected& operator=( const std::unexpected<G>& e ); |
(8) | (since C++23) |
template< class G > constexpr expected& operator=( std::unexpected<G>&& e ); |
(9) | (since C++23) |
Helper function template |
||
template< class T, class U, class... Args > constexpr void reinit-expected( T& newval, U& oldval, Args&&... args ) |
(10) | (since C++23) (exposition only*) |
Assigns a new value to an existing expected
object.
Parameters
other | - | another expected object whose contained value to assign
|
v | - | value to assign to the contained value |
e | - | std::unexpected object whose contained value to assign |
newval | - | the contained value to be constructed |
oldval | - | the contained value to be destroyed |
args | - | the arguments used as initializers of newval |
Effects
Primary template assignment operators
has_value()
and rhs.has_value() have different values (i.e. one of *this and other contains an expected value val
and the other contains an unexpected value unex
), the exposition-only function template reinit-expected
is called to safely update the state.Value ofhas_value()
|
Value of other.has_value() | |
---|---|---|
true | false | |
true | val = *other;
|
reinit-expected (unex , val , other.error());
|
false | reinit-expected (val , unex , *other);
|
unex = other.error();
|
Value ofhas_value()
|
Value of other.has_value() | |
---|---|---|
true | false | |
true | val = std::move(*other);
|
reinit-expected (unex , val , std::move(other.error()));
|
false | reinit-expected (val , unex , std::move(*other));
|
unex = std::move(other.error());
|
has_val
= other.has_value();.Value ofhas_value()
|
Equivalent to |
---|---|
true | val = std::forward<U>(v);
|
false | reinit-expected (val , unex , std::forward<U>(v));has_val = false;
|
Overload | Value ofhas_value()
|
Equivalent to |
---|---|---|
(4) | true | reinit-expected (val , unex , std::forward<const G&>(e.error()));has_val = false;
|
false | unex = std::forward<const G&>(e.error());
| |
(5) | true | reinit-expected (val , unex , std::forward<G>(e.error()));has_val = false;
|
false | unex = std::forward<G>(e.error());
|
void partial specialization assignment operators
Value ofhas_value()
|
Value of other.has_value() | |
---|---|---|
true | false | |
true | (no effects) | std::construct_at (std::addressof(unex ), rhs.unex );has_val = false;
|
false | std::destroy_at(std::addressof(unex ));has_val = true;
|
unex = other.error();
|
Value ofhas_value()
|
Value of other.has_value() | |
---|---|---|
true | false | |
true | (no effects) | std::construct_at (std::addressof(unex ), std::move(rhs.unex ));has_val = false;
|
false | std::destroy_at(std::addressof(unex ));has_val = true;
|
unex = std::move(other.error());
|
Overload | Value ofhas_value()
|
Equivalent to |
---|---|---|
(8) | true | std::construct_at(std::addressof(unex ), std::forward<const G&>(e.error()));has_val = false;
|
false | unex = std::forward<const G&>(e.error());
| |
(9) | true | std::construct_at(std::addressof(unex ), std::forward<G>(e.error()));has_val = false;
|
false | unex = std::forward<G>(e.error());
|
Helper function template
The exposition-only function template reinit-expected
is “defined” as follows:
template<class NewType, class OldType, class... Args> constexpr void reinit-expected(NewType& new_val, OldType& old_val, Args&&... args) { // Case 1: the construction of “new_val” is non-throwing: // “new_val” can be directly constructed after destroying “old_val” if constexpr (std::is_nothrow_constructible_v<NewType, Args...>) { std::destroy_at(std::addressof(old_val)); std::construct_at(std::addressof(new_val), std::forward<Args>(args)...); } // Case 2: the move construction of “new_val” is non-throwing: // constuct a temporary NewType object first // (“old_val” is left intact if an exception is thrown from this construction) else if constexpr (std::is_nothrow_move_constructible_v<NewType>) { NewType temp(std::forward<Args>(args)...); // may throw std::destroy_at(std::addressof(old_val)); std::construct_at(std::addressof(new_val), std::move(temp)); } // Case 3: the construction of “new_val” is potentially-throwing: // a backup of “old_val” is required in order to recover from an exception else { OldType temp(std::move(old_val)); // may throw std::destroy_at(std::addressof(old_val)); try { std::construct_at(std::addressof(new_val), std::forward<Args>(args)...); // may throw } catch (...) { std::construct_at(std::addressof(old_val), std::move(temp)); throw; } } }
This function template is called when the assignment is going to make *this hold the alternative value (i.e. from expected value to unexpected value, or from unexpected value to expected value).
In this case, the old value oldval needs to be destroyed before constructing the new value newval. However, the construction of newval may throw an exception. In order to provide strong exception safety guarantee, the old value needs to be restored before rethrowing the exception so that *this will have a valid state while the exception is being handled.
Return value
Constraints and supplement information
Primary template assignment operators
- std::is_same_v<expected, std::remove_cvref_t<U>> is false.
- std::remove_cvref_t<U> is not a specialization of
std::unexpected
. - All following values are true:
- std::is_constructible_v<E, const G&>
- std::is_assignable_v<E&, const G&>
- std::is_nothrow_constructible_v<E, const G&> || std::is_nothrow_move_constructible_v<T> ||
std::is_nothrow_move_constructible_v<E>
void partial specialization assignment operators
Exceptions
std::is_nothrow_move_constructible_v<T> && std::is_nothrow_move_assignable_v<T> &&
Example
This section is incomplete Reason: no example |
Defect reports
The following behavior-changing defect reports were applied retroactively to previously published C++ standards.
DR | Applied to | Behavior as published | Correct behavior |
---|---|---|---|
LWG 4025 | C++23 | overload (7) was defined as deleted if E is notmove constructible or not move assignable |
it does not participate in overload resolution in this case |
See also
constructs the expected value in-place (public member function) |