• Class / Function / Type

      std::
    • Header file

      <>
    • Other / All

    最終更新日時(UTC):
    が更新

    履歴 編集

    function template
    <algorithm>

    std::unique_copy

    namespace std {
      template <class InputIterator, class OutputIterator>
      OutputIterator
        unique_copy(InputIterator first,
                    InputIterator last,
                    OutputIterator result);    // (1) C++03
    
      template <class InputIterator, class OutputIterator>
      constexpr OutputIterator
        unique_copy(InputIterator first,
                    InputIterator last,
                    OutputIterator result);    // (1) C++20
    
      template <class InputIterator, class OutputIterator,
                class BinaryPredicate>
      OutputIterator
        unique_copy(InputIterator first,
                    InputIterator last,
                    OutputIterator result,
                    BinaryPredicate pred);     // (2) C++03
    
      template <class InputIterator, class OutputIterator,
                class BinaryPredicate>
      constexpr OutputIterator
        unique_copy(InputIterator first,
                    InputIterator last,
                    OutputIterator result,
                    BinaryPredicate pred);     // (2) C++20
    
      template <class ExecutionPolicy, class ForwardIterator1, class ForwardIterator2>
      ForwardIterator2
         unique_copy(ExecutionPolicy&& exec,
                     ForwardIterator1 first,
                     ForwardIterator1 last,
                     ForwardIterator2 result); // (3) C++17
    
      template <class ExecutionPolicy, class ForwardIterator1, class ForwardIterator2,
                class BinaryPredicate>
      ForwardIterator2
        unique_copy(ExecutionPolicy&& exec,
                    ForwardIterator1 first,
                    ForwardIterator1 last,
                    ForwardIterator2 result,
                    BinaryPredicate pred);     // (4) C++17
    }
    

    概要

    イテレータ範囲[first, last)から隣り合った重複要素を取り除き、その結果を出力の範囲へコピーする。

    要件

    • 二項関数オブジェクトpredは、ふたつの値の等値性を判定できなければならない
    • [first,last)[result,result + (last - first)) は重なっていてはならない

    テンプレートパラメータ制約

    • *result = *first は有効な式でなければならない
    • InputIteratorOutputIterator のどちらも forward iterator の要求を満たしていない場合、InputIterator の値型は コピー構築可能 かつ コピー代入可能 でなければならない。そうでない場合は コピー構築可能 であることは要求されない

    効果

    [first,last) 内のイテレータ i について、

    • (1) では *(i - 1) == *i
    • (2) では pred(*(i - 1), *i) != false

    による等値の比較によって連続したグループに分け、それぞれのグループの先頭を result へコピーする。

    戻り値

    結果の範囲の終端を返す。

    計算量

    [first,last) が空の範囲でない場合、正確に last - first - 1 回の比較または述語の適用を行う

    #include <algorithm>
    #include <iostream>
    #include <vector>
    #include <iterator>
    
    void print(const char* tag, const std::vector<int>& v) {
      std::cout << tag << " : ";
      bool first = true;
      for (int x : v) {
        if (first) {
          first = false;
        }
        else {
          std::cout << ',';
        }
        std::cout << x;
      }
      std::cout << std::endl;
    }
    
    int main() {
      // 入力の配列がソート済みではない場合、
      // 隣り合った重複要素が取り除かれる
      {
        std::vector<int> v = { 2,5,3,3,1,2,4,2,1,1,4,4,3,3,3 };
        std::vector<int> uniqued;
    
        // 重複を除いた要素がuniquedに追加されていく
        std::unique_copy(v.begin(), v.end(), std::back_inserter(uniqued));
    
        print("unsorted unique", uniqued);
      }
    
      // 入力の配列がソート済みである場合、
      // 重複している全ての要素が取り除かれて一意になる
      {
        std::vector<int> v = { 2,5,3,3,1,2,4,2,1,1,4,4,3,3,3 };
        std::vector<int> uniqued;
    
        std::sort(v.begin(), v.end());
        std::unique_copy(v.begin(), v.end(), std::back_inserter(uniqued));
    
        print("sorted unique", uniqued);
      }
    }
    

    出力

    unsorted unique : 2,5,3,1,2,4,2,1,4,3
    sorted unique : 1,2,3,4,5
    

    実装例

    template <class InputIterator, class OutputIterator>
    OutputIterator unique_copy(InputIterator first, InputIterator last,
                               OutputIterator result) {
      if (first == last) return result;
    
      auto value = *first++;
      *result++ = value;
      for ( ; first != last; ++first) {
        if (!(value == *first)) {
          value = *first;
          *result++ = value;
        }
      }
    
      return result;
    }
    
    template <class InputIterator, class OutputIterator, class BinaryPredicate>
    OutputIterator unique_copy(InputIterator first, InputIterator last,
                               OutputIterator result, BinaryPredicate pred) {
      if (first == last) return result;
    
      auto value = *first++;
      *result++ = value;
      for ( ; first != last; ++first) {
        if (!pred(value, *first)) {
          value = *first;
          *result++ = value;
        }
      }
    
      return result;
    }
    

    参照