Namespaces
Variants
Views
Actions

std::expected<T,E>::operator=

From cppreference.com
< cpp‎ | utility‎ | expected
 
 
Utilities library
General utilities
Relational operators (deprecated in C++20)
 
 
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.

Contents

[edit] 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

[edit] Effects

[edit] Primary template assignment operators

1,2) Assigns the state of other to *this.
If 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.
1) The contained value is assigned as follows:
Value of
 has_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();
2) The contained value is assigned as follows:
Value of
 has_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());
Then, if no exception was thrown, executes has_val = other.has_value();.
3) The expected value is assigned as follows:
Value of
 has_value() 
Equivalent to
true val = std::forward<U>(v);
false reinit-expected(val, unex, std::forward<U>(v));
has_val = false;
4,5) The unexpected value is assigned as follows:
 Overload  Value of
 has_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());

[edit] void partial specialization assignment operators

6) The unexpected value is assigned or destroyed as follows:
Value of
 has_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();
7) The unexpected value is assigned or destroyed as follows:
Value of
 has_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());
8,9) The unexpected value is assigned as follows:
 Overload  Value of
 has_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());

[edit] 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.

[edit] Return value

1-9) *this

[edit] Constraints and supplement information

[edit] Primary template assignment operators

3) This overload participates in overload resolution only if all following conditions are satisfied:
4) This overload participates in overload resolution only if all following values are true:
5) This overload participates in overload resolution only if all following values are true:

[edit] void partial specialization assignment operators

6) This overload is defined as deleted unless std::is_copy_assignable_v<E> and std::is_copy_constructible_v<E> are both true.
7) This overload participates in overload resolution only if std::is_move_constructible_v<E> and std::is_move_assignable_v<E> are both true.
8) This overload participates in overload resolution only if std::is_constructible_v<E, const G&> and std::is_assignable_v<E&, const G&> are both true.
9) This overload participates in overload resolution only if std::is_constructible_v<E, G> and std::is_assignable_v<E&, G> are both true.

[edit] Exceptions

[edit] Example

[edit] 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 not
move constructible or not move assignable
it does not participate in
overload resolution in this case

[edit] See also

constructs the expected value in-place
(public member function) [edit]