名前空間
変種
操作

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

From cppreference.com
< cpp‎ | algorithm‎ | ranges
 
 
アルゴリズムライブラリ
制約付きアルゴリズムとRangeアルゴリズム (C++20)
制約付きアルゴリズム、例: ranges::copy, ranges::sort, ...
実行ポリシー (C++17)
シーケンスを変更しない操作
一括操作
(C++17)
検索操作
(C++11)                (C++11)(C++11)

シーケンスを変更する操作
コピー操作
(C++11)
(C++11)
スワップ操作
変換操作
生成操作
削除操作
順序変更操作
(C++17まで)(C++11)
(C++20)(C++20)
サンプリング操作
(C++17)

ソートおよび関連操作
パーティション操作
ソート操作
二分探索操作
(パーティション化された範囲)
集合操作 (ソート済み範囲)
マージ操作 (ソート済み範囲)
ヒープ操作
最小/最大操作
(C++11)
(C++17)
辞書順比較操作
順列操作
Cライブラリ
数値演算
未初期化メモリに対する操作
 
制約付きアルゴリズム
このメニューのすべての名前は名前空間 `std::ranges` に属します
シーケンスを変更しない操作
シーケンスを変更する操作
パーティション操作
ソート操作
二分探索操作 (ソート済み範囲)
       
       
集合操作 (ソート済み範囲)
ヒープ操作
最小/最大操作
       
       
順列操作
畳み込み操作
数値演算
(C++23)            
未初期化ストレージに対する操作
戻り値の型
 
ヘッダー <algorithm> で定義
呼び出しシグネチャ
(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 = {} );
(C++23から)
(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 = {} );
(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 = {} );
(C++23から)
(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 = {} );
(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) (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) (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) (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) (C++23から)

特定条件を満たす、範囲 [firstlast) 内の最後の要素を返します。

1) find_lastvalue と等しい要素を検索します。
3) find_last_if は、述語 predtrue を返す、範囲 [firstlast) 内の最後の要素を検索します。
5) find_last_if_not は、述語 predfalse を返す、範囲 [firstlast) 内の最後の要素を検索します。
2,4,6) (1,3,5) と同じですが、r をソース範囲として使用します。これは、ranges::begin(r)first として、ranges::end(r)last として使用するかのようです。

このページで説明されている関数のようなエンティティは、アルゴリズム関数オブジェクト(非公式にはニーブロイドとして知られている)です。つまり、

目次

[編集] パラメータ

first, last - 調査する要素の範囲を定義するイテレータとセンチネルのペア
r - 検査する要素の範囲
value - 要素と比較する値
pred - 射影された要素に適用する述語
proj - 要素に適用する射影

[編集] 戻り値

1,3,5) Etrue となる範囲 [firstlast) 内の最後のイテレータを i とします。
そのようなイテレータが見つかった場合は ranges::subrange<I>{i, last} を返し、見つからなかった場合は ranges::subrange<I>{last, last} を返します。
1) Ebool(std::invoke(proj, *i) == value) です。
3) Ebool(std::invoke(pred, std::invoke(proj, *i))) です。
5) Ebool(!std::invoke(pred, std::invoke(proj, *i))) です。
2,4,6) (1,3,5) と同じですが、戻り値の型は ranges::borrowed_subrange_t<I> です。

[編集] 計算量

述語と射影の適用回数は最大で last - first 回。

[編集] 備考

ranges::find_lastranges::find_last_ifranges::find_last_if_not は、Ibidirectional_iterator または (より良い) random_access_iterator のモデルである場合、一般的な実装においてより効率的です。

機能テストマクロ 規格 機能
__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) アルゴリズム (1,2) のためのリスト初期化

[編集] 可能な実装

これらの実装は、Iforward_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
    {
        // Note: if I is mere forward_iterator, we may only go from begin to end.
        std::optional<I> found;
        for (; first != last; ++first)
            if (std::invoke(proj, *first) == value)
                found = first;
 
        if (!found)
            return {first, first};
 
        return {*found, std::ranges::next(*found, 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
    {
        // Note: if I is mere forward_iterator, we may only go from begin to end.
        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
    {
        // Note: if I is mere forward_iterator, we may only go from begin to end.
        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;

[編集]

#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
    {
        {"one", 1}, {"two", 2}, {"three", 3},
        {"one", 4}, {"two", 5}, {"three", 6},
    };
    auto cmp_one = [](const std::string_view &s) { return s == "one"; };
 
    // find latest element that satisfy the comparator, and projecting pair::first
    const auto subrange = ranges::find_last_if(list, cmp_one, &P::first);
 
    std::cout << "The found element and the tail after it are:\n";
    for (P const& e : subrange)
        std::cout << '{' << std::quoted(e.first) << ", " << e.second << "} ";
    std::cout << '\n';
 
#if __cpp_lib_algorithm_default_value_type
    const auto i3 = ranges::find_last(list, {"three", 3}); // (2) C++26
#else
    const auto i3 = ranges::find_last(list, P{"three", 3}); // (2) C++23
#endif
    assert(i3.begin()->first == "three" && i3.begin()->second == 3);
}

出力

The found element and the tail after it are:
{"one", 4} {"two", 5} {"three", 6}

[編集] 関連項目

特定の範囲内で最後の要素のシーケンスを見つける
(アルゴリズム関数オブジェクト)[編集]
特定の基準を満たす最初の要素を見つける
(アルゴリズム関数オブジェクト)[編集]
要素の範囲の最初の出現を検索する
(アルゴリズム関数オブジェクト)[編集]
あるシーケンスが別のシーケンスの部分シーケンスである場合に true を返す
(アルゴリズム関数オブジェクト)[編集]
部分的に順序付けられた範囲に要素が存在するかどうかを判断する
(アルゴリズム関数オブジェクト)[編集]
範囲が指定された要素または部分範囲を含むかチェックする
(アルゴリズム関数オブジェクト)[編集]
English 日本語 中文(简体) 中文(繁體)