Namespaces
Variants

std::ranges:: find_last, std::ranges:: find_last_if, std::ranges:: find_last_if_not

From cppreference.net
Algorithm library
Constrained algorithms and algorithms on ranges (C++20)
Constrained algorithms, e.g. ranges::copy , ranges::sort , ...
Execution policies (C++17)
Non-modifying sequence operations
Batch operations
(C++17)
Search operations
Modifying sequence operations
Copy operations
(C++11)
(C++11)
Swap operations
Transformation operations
Generation operations
Removing operations
Order-changing operations
(until C++17) (C++11)
(C++20) (C++20)
Sampling operations
(C++17)

Sorting and related operations
Partitioning operations
Sorting operations
Binary search operations
(on partitioned ranges)
Set operations (on sorted ranges)
Merge operations (on sorted ranges)
Heap operations
Minimum/maximum operations
Lexicographical comparison operations
Permutation operations
C library
Numeric operations
Operations on uninitialized memory
Constrained algorithms
All names in this menu belong to namespace std::ranges
Non-modifying sequence operations
Modifying sequence operations
Partitioning operations
Sorting operations
Binary search operations (on sorted ranges)
Set operations (on sorted ranges)
Heap operations
Minimum/maximum operations
Permutation operations
Fold operations
Operations on uninitialized storage
Return types
Definido en el encabezado <algorithm>
Firma de llamada
(1)
template < std:: forward_iterator I, std:: sentinel_for < I > S,

class T,
class Proj = std:: identity >
requires std:: indirect_binary_predicate
< ranges:: equal_to , std :: projected < I, Proj > , const T * >
constexpr ranges:: subrange < I >

find_last ( I first, S last, const T & value, Proj proj = { } ) ;
(desde C++23)
(hasta C++26)
template < std:: forward_iterator I, std:: sentinel_for < I > S,

class Proj = std:: identity ,
class T = std :: projected_value_t < I, Proj > >
requires std:: indirect_binary_predicate
< ranges:: equal_to , std :: projected < I, Proj > , const T * >
constexpr ranges:: subrange < I >

find_last ( I first, S last, const T & value, Proj proj = { } ) ;
(desde C++26)
(2)
template < ranges:: forward_range R,

class T,
class Proj = std:: identity >
requires std:: indirect_binary_predicate
< ranges:: equal_to ,
std :: projected < ranges:: iterator_t < R > , Proj > , const T * >
constexpr ranges:: borrowed_subrange_t < R >

find_last ( R && r, const T & value, Proj proj = { } ) ;
(desde C++23)
(hasta C++26)
template < ranges:: forward_range R,

class Proj = std:: identity ,
class T = std :: projected_value_t < iterator_t < R > , Proj > >
requires std:: indirect_binary_predicate
< ranges:: equal_to ,
std :: projected < ranges:: iterator_t < R > , Proj > , const T * >
constexpr ranges:: borrowed_subrange_t < R >

find_last ( R && r, const T & value, Proj proj = { } ) ;
(desde C++26)
template < std:: forward_iterator I, std:: sentinel_for < I > S,

class Proj = std:: identity ,
std:: indirect_unary_predicate < std :: projected < I, Proj >> Pred >
constexpr ranges:: subrange < I >

find_last_if ( I first, S last, Pred pred, Proj proj = { } ) ;
(3) (desde C++23)
template < ranges:: forward_range R,

class Proj = std:: identity ,
std:: indirect_unary_predicate
< std :: projected < ranges:: iterator_t < R > , Proj >> Pred >
constexpr ranges:: borrowed_subrange_t < R >

find_last_if ( R && r, Pred pred, Proj proj = { } ) ;
(4) (desde C++23)
template < std:: forward_iterator I, std:: sentinel_for < I > S,

class Proj = std:: identity ,
std:: indirect_unary_predicate < std :: projected < I, Proj >> Pred >
constexpr ranges:: subrange < I >

find_last_if_not ( I first, S last, Pred pred, Proj proj = { } ) ;
(5) (desde C++23)
template < ranges:: forward_range R,

class Proj = std:: identity ,
std:: indirect_unary_predicate
< std :: projected < ranges:: iterator_t < R > , Proj >> Pred >
constexpr ranges:: borrowed_subrange_t < R >

find_last_if_not ( R && r, Pred pred, Proj proj = { } ) ;
(6) (desde C++23)

Devuelve el último elemento en el rango [ first , last ) que satisface criterios específicos:

1) find_last busca un elemento igual a value .
3) find_last_if busca el último elemento en el rango [ first , last ) para el cual el predicado pred devuelve true .
5) find_last_if_not busca el último elemento en el rango [ first , last ) para el cual el predicado pred devuelve false .
2,4,6) Igual que (1,3,5) , pero utiliza r como el rango fuente, como si usara ranges:: begin ( r ) como first y ranges:: end ( r ) como last .

Las entidades similares a funciones descritas en esta página son algorithm function objects (conocidas informalmente como niebloids ), es decir:

Contenidos

Parámetros

first, last - el par iterador-centinela que define el rango de elementos a examinar
r - el rango de los elementos a examinar
value - valor contra el cual comparar los elementos
pred - predicado a aplicar a los elementos proyectados
proj - proyección a aplicar a los elementos

Valor de retorno

1,3,5) Sea i el último iterador en el rango [ first , last ) para el cual E es true .
Retorna ranges:: subrange < I > { i, last } , o ranges:: subrange < I > { last, last } si no se encuentra dicho iterador.
1) E es bool ( std:: invoke ( proj, * i ) == value ) .
3) E es bool ( std:: invoke ( pred, std:: invoke ( proj, * i ) ) ) .
5) E es bool ( ! std:: invoke ( pred, std:: invoke ( proj, * i ) ) ) .
2,4,6) Igual que (1,3,5) pero el tipo de retorno es ranges:: borrowed_subrange_t < I > .

Complejidad

Como máximo last - first aplicaciones del predicado y proyección.

Notas

ranges::find_last , ranges::find_last_if , ranges::find_last_if_not tienen mejor eficiencia en implementaciones comunes si I modela bidirectional_iterator o (mejor) random_access_iterator .

Macro de prueba de características Valor Std Característica
__cpp_lib_ranges_find_last 202207L (C++23) ranges::find_last ,
ranges::find_last_if ,
ranges::find_last_if_not
__cpp_lib_algorithm_default_value_type 202403L (C++26) Inicialización de lista para algoritmos ( 1,2 )

Implementación posible

Estas implementaciones solo muestran el algoritmo más lento utilizado cuando I modela forward_iterator .

find_last (1,2)
struct find_last_fn
{
    template<std::forward_iterator I, std::sentinel_for<I> S,
             class Proj = std::identity,
             class T = std::projected_value_t<iterator_t<R>, Proj>>
    requires std::indirect_binary_predicate
                 <ranges::equal_to, std::projected<I, Proj>, const T*>
    constexpr ranges::subrange<I>
        operator()(I first, S last, const T &value, Proj proj = {}) const
    {
        // Nota: si I es solo un forward_iterator, solo podemos ir desde el inicio hasta el final.
        std::optional<I> encontrado;
        for (; first != last; ++first)
            if (std::invoke(proj, *first) == value)
                encontrado = first;
        if (!encontrado)
            return {first, first};
        return {*encontrado, std::ranges::next(*encontrado, last)};
    }
    template<ranges::forward_range R,
             class Proj = std::identity,
             class T = std::projected_value_t<iterator_t<R>, Proj>>
    requires std::indirect_binary_predicate
                 <ranges::equal_to,
                  std::projected<ranges::iterator_t<R>, Proj>, const T*>
    constexpr ranges::borrowed_subrange_t<R>
        operator()(R&& r, const T &value, Proj proj = {}) const
    {
        return this->operator()(ranges::begin(r), ranges::end(r), value, std::ref(proj));
    }
};
inline constexpr find_last_fn find_last;
find_last_if (3,4)
struct find_last_if_fn
{
    template<std::forward_iterator I, std::sentinel_for<I> S,
             class Proj = std::identity,
             std::indirect_unary_predicate<std::projected<I, Proj>> Pred>
    constexpr ranges::subrange<I>
        operator()(I first, S last, Pred pred, Proj proj = {}) const
    {
        // Nota: si I es solo un forward_iterator, solo podemos ir desde el inicio hasta el final.
        std::optional<I> found;
        for (; first != last; ++first)
            if (std::invoke(pred, std::invoke(proj, *first)))
                found = first;
        if (!found)
            return {first, first};
        return {*found, std::ranges::next(*found, last)};
    }
    template<ranges::forward_range R, class Proj = std::identity,
             std::indirect_unary_predicate
                 <std::projected<ranges::iterator_t<R>, Proj>> Pred>
    constexpr ranges::borrowed_subrange_t<R>
        operator()(R&& r, Pred pred, Proj proj = {}) const
    {
        return this->operator()(ranges::begin(r), ranges::end(r),
                                std::ref(pred), std::ref(proj));
    }
};
inline constexpr find_last_if_fn find_last_if;
find_last_if_not (5,6)
struct find_last_if_not_fn
{
    template<std::forward_iterator I, std::sentinel_for<I> S,
             class Proj = std::identity,
             std::indirect_unary_predicate<std::projected<I, Proj>> Pred>
    constexpr ranges::subrange<I>
        operator()(I first, S last, Pred pred, Proj proj = {}) const
    {
        // Nota: si I es solo un forward_iterator, solo podemos ir desde el inicio hasta el final.
        std::optional<I> found;
        for (; first != last; ++first)
            if (!std::invoke(pred, std::invoke(proj, *first)))
                found = first;
        if (!found)
            return {first, first};
        return {*found, std::ranges::next(*found, last)};
    }
    template<ranges::forward_range R, class Proj = std::identity,
             std::indirect_unary_predicate
                 <std::projected<ranges::iterator_t<R>, Proj>> Pred>
    constexpr ranges::borrowed_subrange_t<R>
        operator()(R&& r, Pred pred, Proj proj = {}) const
    {
        return this->operator()(ranges::begin(r), ranges::end(r),
                                std::ref(pred), std::ref(proj));
    }
};
inline constexpr find_last_if_not_fn find_last_if_not;

Ejemplo

#include <algorithm>
#include <cassert>
#include <forward_list>
#include <iomanip>
#include <iostream>
#include <string_view>
int main()
{
    namespace ranges = std::ranges;
    constexpr static auto v = {1, 2, 3, 1, 2, 3, 1, 2};
    {
        constexpr auto i1 = ranges::find_last(v.begin(), v.end(), 3);
        constexpr auto i2 = ranges::find_last(v, 3);
        static_assert(ranges::distance(v.begin(), i1.begin()) == 5);
        static_assert(ranges::distance(v.begin(), i2.begin()) == 5);
    }
    {
        constexpr auto i1 = ranges::find_last(v.begin(), v.end(), -3);
        constexpr auto i2 = ranges::find_last(v, -3);
        static_assert(i1.begin() == v.end());
        static_assert(i2.begin() == v.end());
    }
    auto abs = [](int x) { return x < 0 ? -x : x; };
    {
        auto pred = [](int x) { return x == 3; };
        constexpr auto i1 = ranges::find_last_if(v.begin(), v.end(), pred, abs);
        constexpr auto i2 = ranges::find_last_if(v, pred, abs);
        static_assert(ranges::distance(v.begin(), i1.begin()) == 5);
        static_assert(ranges::distance(v.begin(), i2.begin()) == 5);
    }
    {
        auto pred = [](int x) { return x == -3; };
        constexpr auto i1 = ranges::find_last_if(v.begin(), v.end(), pred, abs);
        constexpr auto i2 = ranges::find_last_if(v, pred, abs);
        static_assert(i1.begin() == v.end());
        static_assert(i2.begin() == v.end());
    }
    {
        auto pred = [](int x) { return x == 1 or x == 2; };
        constexpr auto i1 = ranges::find_last_if_not(v.begin(), v.end(), pred, abs);
        constexpr auto i2 = ranges::find_last_if_not(v, pred, abs);
        static_assert(ranges::distance(v.begin(), i1.begin()) == 5);
        static_assert(ranges::distance(v.begin(), i2.begin()) == 5);
    }
    {
        auto pred = [](int x) { return x == 1 or x == 2 or x == 3; };
        constexpr auto i1 = ranges::find_last_if_not(v.begin(), v.end(), pred, abs);
        constexpr auto i2 = ranges::find_last_if_not(v, pred, abs);
        static_assert(i1.begin() == v.end());
        static_assert(i2.begin() == v.end());
    }
    using P = std::pair<std::string_view, int>;
    std::forward_list<P> list
    {
        {"uno", 1}, {"dos", 2}, {"tres", 3},
        {"uno", 4}, {"dos", 5}, {"tres", 6},
    };
    auto cmp_one = [](const std::string_view &s) { return s == "uno"; };
    // encontrar el último elemento que satisface el comparador, proyectando pair::first
    const auto subrange = ranges::find_last_if(list, cmp_one, &P::primero);
    std::cout << "El elemento encontrado y la cola después de él son:\n";
    for (P const& e : subrange)
        std::cout << '{' << std::quoted(e.primero) << ", " << e.second << "} ";
    std::cout << '\n';
#if __cpp_lib_algorithm_default_value_type
    const auto i3 = ranges::find_last(list, {"tres", 3}); // (2) C++26
#else
    const auto i3 = ranges::find_last(list, P{"tres", 3}); // (2) C++23
#endif
    assert(i3.begin()->first == "tres" && i3.begin()->second == 3);
}

Salida:

El elemento encontrado y la cola después de él son:
{"one", 4} {"two", 5} {"three", 6}

Véase también

encuentra la última secuencia de elementos en un rango determinado
(objeto función de algoritmo)
encuentra el primer elemento que satisface criterios específicos
(objeto función de algoritmo)
busca la primera ocurrencia de un rango de elementos
(objeto función de algoritmo)
retorna true si una secuencia es una subsecuencia de otra
(objeto función de algoritmo)
determina si un elemento existe en un rango parcialmente ordenado
(objeto función de algoritmo)
verifica si el rango contiene el elemento o subrango dado
(objeto función de algoritmo)