名前空間
変種
操作

std::partition_copy

From cppreference.com
< cpp‎ | algorithm
 
 
アルゴリズムライブラリ
制約付きアルゴリズムと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)

ソートおよび関連操作
パーティション操作
partition_copy
(C++11)  
ソート操作
二分探索操作
(パーティション化された範囲)
集合操作 (ソート済み範囲)
マージ操作 (ソート済み範囲)
ヒープ操作
最小/最大操作
(C++11)
(C++17)
辞書順比較操作
順列操作
Cライブラリ
数値演算
未初期化メモリに対する操作
 
ヘッダー <algorithm> で定義
template< class InputIt, class OutputIt1,

          class OutputIt2, class UnaryPred >
std::pair<OutputIt1, OutputIt2>
    partition_copy( InputIt first, InputIt last,
                    OutputIt1 d_first_true, OutputIt2 d_first_false,

                    UnaryPred p );
(1) (C++11以降)
(C++20 以降 constexpr)
template< class ExecutionPolicy, class ForwardIt1, class ForwardIt2,

          class ForwardIt3, class UnaryPred >
std::pair<ForwardIt2, ForwardIt3>
    partition_copy( ExecutionPolicy&& policy,
                    ForwardIt1 first, ForwardIt1 last,
                    ForwardIt2 d_first_true, ForwardIt3 d_first_false,

                    UnaryPred p );
(2) (C++17以降)
1) 述語 p が返す値に応じて、範囲 [firstlast) から要素を2つの異なる範囲にコピーします。
  • 述語 p を満たす要素は、d_first_true で始まる範囲にコピーされます。
  • 残りの要素は、d_first_false で始まる範囲にコピーされます。
2) (1) と同じですが、policy に従って実行されます。
このオーバーロードは、以下のすべての条件が満たされた場合にのみオーバーロード解決に参加します。

std::is_execution_policy_v<std::decay_t<ExecutionPolicy>>true である。

(C++20まで)

std::is_execution_policy_v<std::remove_cvref_t<ExecutionPolicy>>true です。

(C++20以降)

*firstd_first_true または d_first_false書き込み可能でない場合、プログラムは不正です。

入力範囲と2つの出力範囲の間で、いずれかの2つの範囲が重複する場合、動作は未定義です。

目次

[編集] パラメータ

first, last - コピー元の要素のソース範囲を定義するイテレータのペア
d_first_true - p を満たす要素の出力範囲の開始
d_first_false - p を満たさない要素の出力範囲の開始
policy - 使用する 実行ポリシー
p - 要素が d_first_true に配置されるべき場合、true を返す単項述語。

p(v) は、型が (const の可能性もある) VT であるすべての引数 v に対して bool に変換可能でなければなりません。ここで VTInputIt の値型であり、値カテゴリに関わらず、v を変更してはなりません。したがって、パラメータ型として VT& は許可されません。また、VT のムーブがコピーと同等である場合を除き、VT も許可されません(C++11以降)

型要件
-
InputItLegacyInputIterator の要件を満たす必要があります。
-
OutputIt1, OutputIt2LegacyOutputIterator の要件を満たしている必要があります。
-
ForwardIt1, ForwardIt2, ForwardIt3LegacyForwardIterator の要件を満たしている必要があります。
-
UnaryPredPredicate の要件を満たさなければなりません。

[編集] 戻り値

d_first_true 範囲の終端イテレータと d_first_false 範囲の終端イテレータから構築された std::pair

[編集] 計算量

正確に std::distance(first, last) 回の p の適用。

オーバーロード (2) の場合、ForwardIt の値型が CopyConstructible でない場合、パフォーマンスコストが発生する可能性があります。

[編集] 例外

テンプレートパラメータ ExecutionPolicy を持つオーバーロードは、次のようにエラーを報告します。

  • アルゴリズムの一部として呼び出された関数の実行が例外をスローし、ExecutionPolicy標準ポリシー のいずれかである場合、std::terminate が呼び出されます。その他の ExecutionPolicy の場合、動作は実装定義です。
  • アルゴリズムがメモリの割り当てに失敗した場合、std::bad_alloc がスローされます。

[編集] 可能な実装

partition_copy (1)
template<class InputIt, class OutputIt1,
         class OutputIt2, class UnaryPred>
constexpr //< since C++20
std::pair<OutputIt1, OutputIt2>
    partition_copy(InputIt first, InputIt last,
                   OutputIt1 d_first_true, OutputIt2 d_first_false,
                   UnaryPred p)
{
    for (; first != last; ++first)
    {
        if (p(*first))
        {
            *d_first_true = *first;
            ++d_first_true;
        }
        else
        {
            *d_first_false = *first;
            ++d_first_false;
        }
    }
 
    return std::pair<OutputIt1, OutputIt2>(d_first_true, d_first_false);
}

[編集]

#include <algorithm>
#include <iostream>
#include <utility>
 
void print(auto rem, const auto& v)
{
    for (std::cout << rem; const auto& x : v)
        std::cout << x << ' ';
    std::cout << '\n';
}
 
int main()
{
    int arr[10] = {0, 1, 2, 3, 4, 5, 6, 7, 8, 9};
    int true_arr[5] = {0};
    int false_arr[5] = {0};
 
    std::partition_copy(std::begin(arr), std::end(arr),
                        std::begin(true_arr), std::begin(false_arr),
                        [](int i) { return 4 < i; });
 
    print("true_arr:  ", true_arr);
    print("false_arr: ", false_arr);
}

出力

true_arr:  5 6 7 8 9
false_arr: 0 1 2 3 4

[編集] 欠陥報告

以下の動作変更を伴う欠陥報告が、以前に公開されたC++標準に遡って適用されました。

DR 適用対象 公開された動作 正しい動作
P0896R4 C++11
C++17
1. InputIt (C++11)/ForwardIt1 (C++17) の値型
    は CopyAssignable である必要があった
2. 2つの出力範囲は重複する可能性がある
1. 必要ない
2. この場合の動作は
    未定義

[編集] 関連項目

要素の範囲を2つのグループに分割する
(関数テンプレート) [編集]
相対的な順序を維持しながら要素を2つのグループに分割する
(関数テンプレート) [編集]
要素の範囲を新しい場所にコピーする
(関数テンプレート) [編集]
特定の基準を満たす要素を除外して範囲をコピーする
(関数テンプレート) [編集]
要素を2つのグループに分割しながら範囲をコピーする
(アルゴリズム関数オブジェクト)[編集]
English 日本語 中文(简体) 中文(繁體)