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

履歴 編集

function template
<algorithm>

std::unique

namespace std {
  template <class ForwardIterator>
  ForwardIterator
    unique(ForwardIterator first,
           ForwardIterator last); // (1) C++03

  template <class ForwardIterator>
  constexpr ForwardIterator
    unique(ForwardIterator first,
           ForwardIterator last); // (1) C++20

  template <class ForwardIterator, class BinaryPredicate>
  ForwardIterator
    unique(ForwardIterator first,
           ForwardIterator last,
           BinaryPredicate pred); // (2) C++03

  template <class ForwardIterator, class BinaryPredicate>
  constexpr ForwardIterator
    unique(ForwardIterator first,
           ForwardIterator last,
           BinaryPredicate pred); // (2) C++20

  template <class ExecutionPolicy, class ForwardIterator>
  ForwardIterator
    unique(ExecutionPolicy&& exec,
           ForwardIterator first,
           ForwardIterator last); // (3) C++17

  template <class ExecutionPolicy, class ForwardIterator, class BinaryPredicate>
  ForwardIterator
    unique(ExecutionPolicy&& exec,
           ForwardIterator first,
           ForwardIterator last,
           BinaryPredicate pred);  // (4) C++17
}

概要

イテレータ範囲[first, last)から重複した要素を除ける。

この関数は、隣り合った重複要素を除いた要素を、範囲の先頭に集める。この関数によってコンテナから直接要素が削除され、コンテナの要素数が減るようなことはない。コンテナから実際に要素を削除する場合は、この関数の戻り値として、先頭に集められた重複なし範囲の末尾の次を指すイテレータが返るため、そのイテレータを介してコンテナのerase()メンバ関数などを呼び出し、削除を行うこと。

この関数の戻り値として返されるイテレータ以降の値は未規定

要件

  • 二項関数オブジェクトpredは、ふたつの値の等値性を判定できなければならない

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

  • *firstの型がMoveAssignable要件を満たすこと

効果

[first,last) が空の範囲でない場合、[first + 1,last) 内のイテレータ i について、

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

による等値の比較によって連続したグループに分け、それぞれのグループの先頭以外を取り除く。

戻り値

重複を除いた範囲の、末尾の次を指すイテレータを返す

計算量

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

例 (C++11)

#include <algorithm>
#include <iostream>
#include <vector>

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 };

    decltype(v)::iterator result = std::unique(v.begin(), v.end());

    // [v.begin(), result)の範囲に、重複を除いた結果が入っている。
    // 不要になった要素を削除
    v.erase(result, v.end());

    print("unsorted unique", v);
  }

  // 入力の配列がソート済みである場合、
  // 重複している全ての要素が取り除かれて一意になる
  {
    std::vector<int> v = { 2,5,3,3,1,2,4,2,1,1,4,4,3,3,3 };

    std::sort(v.begin(), v.end());
    decltype(v)::iterator result = std::unique(v.begin(), v.end());

    // 不要になった要素を削除
    v.erase(result, v.end());

    print("sorted unique", v);
  }
}

出力

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

実装例

template <class ForwardIterator>
ForwardIterator unique(ForwardIterator first, ForwardIterator last) {
  if (first == last) return first;

  auto result = first;
  auto value = move(*first++);
  for ( ; first != last; ++first) {
    if (!(value == *first)) {
      *result++ = move(value);
      value = move(*first);
    }
  }
  *result++ = move(value);

  return result;
}

template <class ForwardIterator, class BinaryPredicate>
ForwardIterator unique(ForwardIterator first, ForwardIterator last, BinaryPredicate pred) {
  if (first == last) return first;

  auto result = first;
  auto value = move(*first++);
  for ( ; first != last; ++first) {
    if (!pred(value, *first)) {
      *result++ = move(value);
      value = move(*first);
    }
  }
  *result++ = move(value);

  return result;
}

参照