Пространства имён
Варианты
Действия

std::optional<T>::and_then

Материал из cppreference.com
< cpp‎ | utility‎ | optional
 
 
Библиотека утилит
Языковая поддержка
Поддержка типов (базовые типы, RTTI)
Макросы тестирования функциональности библиотеки (C++20)    
Управление динамической памятью
Программные утилиты
Поддержка сопрограмм (C++20)
Вариативные функции
Трёхстороннее сравнение (C++20)
(C++20)
(C++20)(C++20)(C++20)(C++20)(C++20)(C++20)
Общие утилиты
Дата и время
Функциональные объекты
Библиотека форматирования (C++20)
(C++11)
Операторы отношения (устарело в C++20)
Целочисленные функции сравнения
(C++20)(C++20)(C++20)    
(C++20)
Операции обмена и типа
(C++14)
(C++11)
(C++11)
(C++11)
(C++17)
Общие лексические типы
(C++11)
(C++17)
(C++17)
(C++17)
(C++11)
(C++17)
(C++23)
Элементарные преобразования строк
(C++17)
(C++17)
 
std::optional
Функции-элементы
Наблюдатели
Монадические операции
optional::and_then
(C++23)
Модификаторы
Функции, не являющиеся элементами
Правила вывода
Вспомогательные классы
Вспомогательные объекты
 
template< class F >
constexpr auto and_then( F&& f ) &;
(1) (начиная с C++23)
template< class F >
constexpr auto and_then( F&& f ) const&;
(2) (начиная с C++23)
template< class F >
constexpr auto and_then( F&& f ) &&;
(3) (начиная с C++23)
template< class F >
constexpr auto and_then( F&& f ) const&&;
(4) (начиная с C++23)

Если *this содержит значение, вызывает f с содержащимся значением в качестве аргумента и возвращает результат этого вызова; в противном случае возвращает пустой std::optional.

Тип возвращаемого значения (смотрите ниже) должен быть специализацией std::optional (в отличие от transform()). Иначе программа некорректна.

1) Эквивалентно
if (*this)
    return std::invoke(std::forward<F>(f), **this);
else
    return std::remove_cvref_t<std::invoke_result_t<F, T&>>{};
2) Эквивалентно
if (*this)
    return std::invoke(std::forward<F>(f), **this);
else
    return std::remove_cvref_t<std::invoke_result_t<F, const T&>>{};
3) Эквивалентно
if (*this)
    return std::invoke(std::forward<F>(f), std::move(**this));
else
    return std::remove_cvref_t<std::invoke_result_t<F, T>>{};
4) Эквивалентно
if (*this)
    return std::invoke(std::forward<F>(f), std::move(**this));
else
    return std::remove_cvref_t<std::invoke_result_t<F, const T>>{};

Содержание

[править] Параметры

f подходящая функция или объект Callable, который возвращает std::optional

[править] Возвращаемое значение

Результат f или пустой std::optional, как описано выше.

[править] Примечание

Некоторые языки называют эту операцию плоской картой.

Макрос Тестирования функциональности Значение Стандарт Функциональность
__cpp_lib_optional 202110L (C++23) Монадические операции в std::optional

[править] Пример

#include <charconv>
#include <iomanip>
#include <iostream>
#include <optional>
#include <ranges>
#include <string>
#include <string_view>
#include <vector>
 
std::optional<int> to_int(std::string_view sv)
{
    int r {};
    auto [ptr, ec] { std::from_chars(sv.data(), sv.data() + sv.size(), r) };
    if (ec == std::errc())
        return r;
    else
        return std::nullopt;
}
 
int main()
{
    using namespace std::literals;
 
    const std::vector<std::optional<std::string>> v
    {
        "1234", "15 foo", "bar", "42", "5000000000", " 5", std::nullopt, "-43"
    };
 
    for (auto&& x : v | std::views::transform(
        [](auto&& o)
        {
            // отладочная печать содержимого ввода optional<string>
            std::cout << std::left << std::setw(13)
                      << std::quoted(o.value_or("nullopt")) << " -> ";
 
            return o
                // если optional равен nullopt, преобразовать его в optional с строкой ""
                .or_else([]{ return std::optional{""s}; })
                // плоская карта из строк в int (создание пустых optional там,
                // где это не удаётся)
                .and_then(to_int)
                // отображение int на int + 1
                .transform([](int n) { return n + 1; })
                // преобразовать обратно в строки
                .transform([](int n) { return std::to_string(n); })
                // заменить все пустые optional, оставленные
                // and_then и проигнорировать преобразованием с "NaN"
                .value_or("NaN"s);
        }))
        std::cout << x << '\n';
}

Вывод:

"1234"        -> 1235
"15 foo"      -> 16
"bar"         -> NaN
"42"          -> 43
"5000000000"  -> NaN
" 5"          -> NaN
"nullopt"     -> NaN
"-43"         -> -42

[править] Смотрите также

возвращает содержащееся значение, если доступно, иначе другое значение
(public функция-элемент) [править]
(C++23)
возвращает optional, содержащий преобразованное содержащееся значение, если оно существует, или пустой optional в противном случае
(public функция-элемент) [править]
(C++23)
возвращает сам optional, если он содержит значение, или результат данной функции иначе
(public функция-элемент) [править]