std::ranges::partition_copy, std::ranges::partition_copy_result

来自cppreference.com
< cpp‎ | algorithm‎ | ranges
 
 
算法库
受约束算法及范围上的算法 (C++20)
受约束算法: std::ranges::copy, std::ranges::sort, ...
执行策略 (C++17)
不修改序列的操作
(C++11)(C++11)(C++11)
(C++17)
修改序列的操作
未初始化存储上的操作
划分操作
排序操作
(C++11)
二分搜索操作
集合操作(在已排序范围上)
堆操作
(C++11)
最小/最大操作
(C++11)
(C++17)

排列
数值运算
C 库
 
受约束算法
不修改序列的操作
修改序列的操作
划分操作
ranges::partition_copy
排序操作
二分搜索操作
集合操作(在已排序范围上)
堆操作
最小/最大操作
排列
未初始化存储上的操作
返回类型
 
在标头 <algorithm> 定义
调用签名
template< std::input_iterator I, std::sentinel_for<I> S,

          std::weakly_incrementable O1, std::weakly_incrementable O2,
          class Proj = std::identity, std::indirect_unary_predicate<
          std::projected<I, Proj>> Pred >
requires  std::indirectly_copyable<I, O1> && std::indirectly_copyable<I, O2>
constexpr partition_copy_result<I, O1, O2>

          partition_copy( I first, S last, O1 out_true, O2 out_false, Pred pred, Proj proj = {} );
(1) (C++20 起)
template< ranges::input_range R, std::weakly_incrementable O1, std::weakly_incrementable O2,

          class Proj = std::identity,
          std::indirect_unary_predicate<std::projected<iterator_t<R>, Proj>> Pred >
requires  std::indirectly_copyable<ranges::iterator_t<R>, O1> &&
          std::indirectly_copyable<ranges::iterator_t<R>, O2>
constexpr partition_copy_result<ranges::borrowed_iterator_t<R>, O1, O2>

          partition_copy( R&& r, O1 out_true, O2 out_false, Pred pred, Proj proj = {} );
(2) (C++20 起)
辅助类型
template<class I, class O1, class O2>
using partition_copy_result = ranges::in_out_out_result<I, O1, O2>;
(3) (C++20 起)
1) 从输入范围 [first, last) 复制元素到二个不同的取决于谓词 pred 所返回的值输出范围。复制在由 proj 投影后满足 pred 的元素到始于 out_true 的范围。复制剩余元素到始于 out_false 的范围。
前条件:输入范围不与任一输出范围重叠。
2)(1) ,但以 r 为源范围,如同以 ranges::begin(r)first 并以 ranges::end(r)last

此页面上描述的仿函数实体是 niebloid ,即:

实际上,它们能以函数对象,或以某些特殊编译器扩展实现。

参数

first, last - 要自之复制的输入元素范围
r - 要自之复制的输入元素范围
out_true - 满足 pred 的输出元素范围的起始
out_false - 不满足 pred 的输出元素范围的起始
pred - 应用到投影后元素的谓词
proj - 应用到元素的投影

返回值

等于 {last, o1, o2} 的对象,其中 o1o2 分别为复制完成后两个输出范围的末尾。

复杂度

准确应用 ranges::distance(first, last) 次对应的谓词 comp 和任何投影 proj

可能的实现

struct partition_copy_fn {
  template <std::input_iterator I, std::sentinel_for<I> S,
           std::weakly_incrementable O1, std::weakly_incrementable O2,
           class Proj = std::identity, std::indirect_unary_predicate<
           std::projected<I, Proj>> Pred>
  requires std::indirectly_copyable<I, O1> && std::indirectly_copyable<I, O2>
  constexpr ranges::partition_copy_result<I, O1, O2>
  operator()( I first, S last, O1 out_true, O2 out_false, Pred pred, Proj proj = {} ) const {
      for (; first != last; ++first)
          if (!!std::invoke(pred, std::invoke(proj, *first)))
              *out_true = *first, ++out_true;
          else
              *out_false = *first, ++out_false;
      return {std::move(first), std::move(out_true), std::move(out_false)};
  }
 
  template<ranges::input_range R, std::weakly_incrementable O1, std::weakly_incrementable O2,
           class Proj = std::identity,
           std::indirect_unary_predicate<std::projected<iterator_t<R>, Proj>> Pred>
  requires std::indirectly_copyable<ranges::iterator_t<R>, O1> &&
           std::indirectly_copyable<ranges::iterator_t<R>, O2>
  constexpr ranges::partition_copy_result<ranges::borrowed_iterator_t<R>, O1, O2>
  operator()( R&& r, O1 out_true, O2 out_false, Pred pred, Proj proj = {} ) const {
      return (*this)(ranges::begin(r), ranges::end(r), std::move(out_true),
                     std::move(out_false), std::move(pred), std::move(proj));
  }
};
 
inline constexpr partition_copy_fn partition_copy{};

示例

#include <algorithm>
#include <cctype>
#include <iostream>
#include <vector>
#include <iterator>
 
int main()
{
    const auto in = {'N', '3', 'U', 'M', '1', 'B', '4', 'E', '1', '5', 'R', '9'};
 
    std::vector<int> o1(size(in)), o2(size(in));
 
    auto pred = [](char c){ return std::isalpha(c); };
 
    auto ret = std::ranges::partition_copy(in, o1.begin(), o2.begin(), pred);
 
    std::ostream_iterator<char> cout {std::cout, " "};
    std::cout << "in = ";
    std::ranges::copy(in, cout);
    std::cout << "\no1 = ";
    std::copy(o1.begin(), ret.out1, cout);
    std::cout << "\no2 = ";
    std::copy(o2.begin(), ret.out2, cout);
    std::cout << '\n';
}

输出:

in = N 3 U M 1 B 4 E 1 5 R 9
o1 = N U M B E R
o2 = 3 1 4 1 5 9

参阅

将范围中的元素分为二组
(niebloid)
将元素分成二组,同时保持其相对顺序
(niebloid)
将某一范围的元素复制到一个新的位置
(niebloid)
复制一个范围的元素,忽略满足特定判别标准的元素
(niebloid)
复制一个范围,将各元素分为两组
(函数模板)