2 Algorithm

2.1 std::transform

  • std::transform applies the given function to the elements of the given input range(s), and stores the result in an output range starting from d_first.

parameters

first1, last1: the pair of iterators defining the source range of elements to transform ;  
first2: the beginning of the second range of elements to transform, (3,4) only;  
d_first: the beginning of the destination range, may be equal to first1 or first2;  
policy: the execution policy to use;  
unary_op: Ret fun(const Type &a);  
binary_op: Ret fun(const Type1 &a, const Type2 &b);  

possible implementation

template<class InputIt, class OutputIt, class UnaryOp>
constexpr //< since C++20
OutputIt transform(InputIt first1, InputIt last1,
                   OutputIt d_first, UnaryOp unary_op)
{
    for (; first1 != last1; ++d_first, ++first1)
        *d_first = unary_op(*first1);
 
    return d_first;
}

template<class InputIt1, class InputIt2, 
         class OutputIt, class BinaryOp>
constexpr //< since C++20
OutputIt transform(InputIt1 first1, InputIt1 last1, InputIt2 first2,
                   OutputIt d_first, BinaryOp binary_op)
{
    for (; first1 != last1; ++d_first, ++first1, ++first2)
        *d_first = binary_op(*first1, *first2);
 
    return d_first;
}

example

#include <algorithm>
#include <cctype>
#include <iomanip>
#include <iostream>
#include <string>
#include <utility>
#include <vector>
 
void print_ordinals(const std::vector<unsigned>& ordinals)
{
    std::cout << "ordinals: ";
    for (unsigned ord : ordinals)
        std::cout << std::setw(3) << ord << ' ';
    std::cout << '\n';
}
 
char to_uppercase(unsigned char c)
{
    return std::toupper(c);
}
 
void to_uppercase_inplace(char& c)
{
    c = to_uppercase(c);
}
 
void unary_transform_example(std::string& hello, std::string world)
{
    // Transform string to uppercase in-place
 
    std::transform(hello.cbegin(), hello.cend(), hello.begin(), to_uppercase);
    std::cout << "hello = " << std::quoted(hello) << '\n';
 
    // for_each version (see Notes above)
    std::for_each(world.begin(), world.end(), to_uppercase_inplace);
    std::cout << "world = " << std::quoted(world) << '\n';
}
 
void binary_transform_example(std::vector<unsigned> ordinals)
{
    // Transform numbers to doubled values
 
    print_ordinals(ordinals);
 
    std::transform(ordinals.cbegin(), ordinals.cend(), ordinals.cbegin(),
                   ordinals.begin(), std::plus<>{});
 
    print_ordinals(ordinals);
}
 
int main()
{
    std::string hello("hello");
    unary_transform_example(hello, "world");
 
    std::vector<unsigned> ordinals;
    std::copy(hello.cbegin(), hello.cend(), std::back_inserter(ordinals));
    binary_transform_example(std::move(ordinals));
}
// OUTPUT
// hello = "HELLO"
// world = "WORLD"
// ordinals:  72  69  76  76  79 
// ordinals: 144 138 152 152 158

2.2 std::accumulate

Computes the sum of the given value init and the elements in the range [first, last).

parameters

  • first, last - the pair of iterators defining the range of elements to accumulate
  • init - initial value of the accumulate
  • op - Ret fun(const Type1 &a, const Type2 &b);
    如标准库中的std::plus<>
    

possible implementation


accumulate (1)
template<class InputIt, class T>
constexpr // since C++20
T accumulate(InputIt first, InputIt last, T init)
{
    for (; first != last; ++first)
        init = std::move(init) + *first; // std::move since C++20
 
    return init;
}

template<class InputIt, class T, class BinaryOperation>
constexpr // since C++20
T accumulate(InputIt first, InputIt last, T init, BinaryOperation op)
{
    for (; first != last; ++first)
        init = op(std::move(init), *first); // std::move since C++20
 
    return init;
}

2.3 std::optinal

  • std::optional最高效的写法是触发RVO的写法,即:

    optional<A> optional_best(int n) {
        optional<A> temp(someFn(n));
        return temp;
    }
    
    

2.4 std::move

  • [first, last) 范围内的元素移动到以 d_first 开始的目标范围
template <typename InputIt, typename OutputIt>
OutputIt std::move(InputIt first, InputIt last, OutputIt d_first);
  • 返回一个迭代器,指向目标范围移动后的结束位置(即 d_first + (last - first))。
参数类型描述
firstInputIt源范围的起始迭代器(指向要移动的第一个元素)
lastInputIt源范围的结束迭代器(指向最后一个元素的下一个位置)
d_firstOutputIt目标范围的起始迭代器(指向移动后第一个元素位置)

2.5 std::move_backward

  • 目标位置:元素会被移动到以 d_last 为结束的目标范围,即目标范围是 [d_last - N, d_last),其中 N = last - first。
template <class BidirIt1, class BidirIt2>
BidirIt2 std::move_backward(BidirIt1 first, BidirIt1 last, BidirIt2 d_last);
  • 返回一个迭代器,指向目标范围移动后的起始位置(即 d_last - (last - first))
参数类型描述
firstBidirIt1源范围的起始迭代器(指向要移动的第一个元素)
lastBidirIt1源范围的结束迭代器(指向最后一个元素的下一个位置)
d_lastBidirIt2目标范围的结束迭代器(指向移动后最后一个元素的下一个位置)

2.6 std::lower_bound 和 std::upper_bound

  • 实际上是二分查找的实现
std::lower_bound: 返回第一个不小于给定值的元素位置。
返回值: 指向第一个满足 *it >= value 的元素的迭代器。若没有这样的元素,则返回 end()
std::upper_bound: 返回第一个大于给定值的元素位置。
返回值: 指向第一个满足 *it > value 的元素的迭代器。若没有这样的元素,则返回 end()。

2.7 std::distance

std::distance 返回从第一个迭代器到第二个迭代器之间的元素数量。对于随机访问迭代器(如 std::vector、std::deque、std::array 的迭代器),它的时间复杂度是 O(1);对于其他类型的迭代器(如 std::list、std::forward_list 的迭代器),时间复杂度是 O(n),因为它需要逐个遍历元素。
template<class InputIterator>
typename std::iterator_traits<InputIterator>::difference_type
    std::distance(InputIterator first, InputIterator last);

2.8 std::all_of, std::any_of, std::none_of

std::all_of: 检查所有元素是否都满足条件(Predicate 返回 true)。  
std::any_of: 检查是否至少有一个元素满足条件。  
std::non_of: 检查是否全部不满足条件
  • 参数说明 |参数| 说明| |---|---| |first| 起始迭代器,指向待检查范围的第一个元素。| |last| 终止迭代器,指向待检查范围的末尾(最后一个元素的下一个位置)。| |p| 谓词(Predicate),接受一个元素类型的参数,返回 bool 类型的条件结果。|

2.9 std::count, std::count_if

  • std::count

    统计值为 value 的元素个数
    
    template<class InputIt, class T>
    typename iterator_traits<InputIt>::difference_type
          count(InputIt first, InputIt last, const T& value);
    
  • std::count_if

    统计符合条件的元素个数
    
    template<class InputIt, class Predicate>
    typename iterator_traits<InputIt>::difference_type
      count_if(InputIt first, InputIt last, Predicate p);
    

2.10 std::find, std::find_if

  • std::find: 返回第一个等于 value 的迭代器

    自定义对象使用std::find,需重载 operator==
    
    template<class InputIt, class T>
    InputIt find(InputIt first, InputIt last, const T& value);
    
  • std::find_if: 在范围内查找第一个满足谓词条件的元素。

    template<class InputIt, class Predicate>
    InputIt find_if(InputIt first, InputIt last, Predicate p);
    

2.11 std::copy, std::copy_if

  • std::copy: 完全复制所有元素

    template<class InputIt, class OutputIt, class Predicate>
    OutputIt copy(InputIt first, InputIt last, OutputIt d_first);
    
  • std::copy_if: 选择性复制元素

    template<class InputIt, class OutputIt, class Predicate>
    OutputIt copy_if(InputIt first, InputIt last, OutputIt d_first, Predicate pred);
    

2.12 std::fill, std::generate

  • std::fill: 用固定值填充范围
    template<class ForwardIt, class T>
    void fill(ForwardIt first, ForwardIt last, const T& value);
    
  • std::generate: 用生成器函数填充范围
    template<class ForwardIt, class Generator>
    void generate(ForwardIt first, ForwardIt last, Generator gen);
    
可以使用随机数生成器来进行填充
#include <vector>
#include <algorithm>
#include <random>

std::vector<int> v(5);
int counter = 0;
std::generate(v.begin(), v.end(), [&counter]() {
    return counter++; // 生成 0, 1, 2, 3, 4
});

// 生成随机数
std::random_device rd;
std::mt19937 rng(rd());
std::uniform_int_distribution<int> dist(1, 10);
std::generate(v.begin(), v.end(), [&]() { return dist(rng); });

2.13 std::search, std::mismatch

  • std::search: 在序列中搜索子序列。

      template<class ForwardIt1, class ForwardIt2>
      ForwardIt1 search(ForwardIt1 first1, ForwardIt1 last1, ForwardIt2 first2, ForwardIt2 last2);
    
      // 示例:在 v1 中查找子序列 v2
      std::vector<int> v1 = {1, 2, 3, 4, 5, 6};
      std::vector<int> v2 = {3, 4};
      auto it = std::search(v1.begin(), v1.end(), v2.begin(), v2.end()); // 返回指向3的迭代器
    
  • std::mismatch: 在比较两个序列,返回第一个不匹配的位置。

      template<class InputIt1, class InputIt2>
      std::pair<InputIt1, InputIt2> mismatch(InputIt1 first1, InputIt1 last1, InputIt2 first2);
    
      // 示例:比较两个字符串
      std::string s1 = "hello", s2 = "hxllo";
      auto [it1, it2] = std::mismatch(s1.begin(), s1.end(), s2.begin()); 
      // it1指向s1的'e', it2指向s2的'x'
    

2.14 std::replace, std::replace_if

  • std::replace: 替换所有等于 old_value 的元素。
template<class ForwardIt, class T>
void replace(ForwardIt first, ForwardIt last, const T& old_value, const T& new_value);

// 示例:替换所有3为5
std::vector<int> v = {1, 2, 3, 3, 4};
std::replace(v.begin(), v.end(), 3, 5); // v变为{1, 2, 5, 5, 4}
  • std::replace_if: 替换满足谓词的元素
template<class ForwardIt, class Predicate, class T>
void replace_if(ForwardIt first, ForwardIt last, Predicate pred, const T& new_value);

// 示例:替换所有偶数为0
std::replace_if(v.begin(), v.end(), [](int x) { return x % 2 == 0; }, 0);

2.15 std::remove / std::remove_if

移动满足条件的元素到末尾,返回新逻辑结尾。
template<class ForwardIt, class T>
ForwardIt remove(ForwardIt first, ForwardIt last, const T& value);

template<class ForwardIt, class Predicate>
ForwardIt remove_if(ForwardIt first, ForwardIt last, Predicate pred);

// 示例:删除所有3(需结合erase)
std::vector<int> v = {1, 2, 3, 4, 3};
auto new_end = std::remove(v.begin(), v.end(), 3);
v.erase(new_end, v.end()); // v变为{1, 2, 4}

2.16 std::reverse

  • 反转整个序列
template<class BidirIt>
void reverse(BidirIt first, BidirIt last);

// 示例:反转vector
std::reverse(v.begin(), v.end()); // {1, 2, 3} → {3, 2, 1}

2.17 std::rotate

  • 把middle旋转到开头
template<class ForwardIt>
ForwardIt rotate(ForwardIt first, ForwardIt middle, ForwardIt last);

// 示例:将中间元素旋转到开头
std::vector<int> v = {1, 2, 3, 4, 5};
std::rotate(v.begin(), v.begin() + 2, v.end()); // v变为{3, 4, 5, 1, 2}

2.18 std::shuffle

  • 使用一个随机生成器,打乱序列
template<class RandomIt, class RandomGen>
void shuffle(RandomIt first, RandomIt last, RandomGen&& g);

// 示例:使用随机引擎
#include <random>
std::vector<int> v = {1, 2, 3, 4};
std::random_device rd;
std::mt19937 rng(rd());
std::shuffle(v.begin(), v.end(), rng); // 随机排列

2.19 std::unique

删除相邻重复元素(需先排序)。
template<class ForwardIt>
ForwardIt unique(ForwardIt first, ForwardIt last);

// 示例:删除相邻重复项
std::vector<int> v = {1, 1, 2, 3, 3};
auto last = std::unique(v.begin(), v.end());
v.erase(last, v.end()); // v变为{1, 2, 3}

2.20 std::sort, std::stable_sort, std::partial_sort

  • std::partial_sort是部分排序
template<class RandomIt>
void sort(RandomIt first, RandomIt last);

template<class RandomIt>
void stable_sort(RandomIt first, RandomIt last);

// 示例:降序排序
std::sort(v.begin(), v.end(), [](int a, int b) { return a > b; });

template<class RandomIt>
void partial_sort(RandomIt first, RandomIt middle, RandomIt last);

// 示例:找到前3小的元素
std::vector<int> v = {5, 3, 1, 4, 2};
std::partial_sort(v.begin(), v.begin() + 3, v.end()); // 前3个元素为{1, 2, 3}

2.21 std::nth_element

  • 将第n个元素排序到正确的位置上
template<class RandomIt>
void nth_element(RandomIt first, RandomIt nth, RandomIt last);

// 示例:找到第3小的元素
std::nth_element(v.begin(), v.begin() + 2, v.end()); 
// v = 3,其他元素相对无序

2.22 std::merge

  • 合并两个有序序列
template<class InputIt1, class InputIt2, class OutputIt>
OutputIt merge(InputIt1 first1, InputIt1 last1, InputIt2 first2, InputIt2 last2, OutputIt d_first);

// 示例:合并两个有序vector
std::vector<int> v1 = {1, 3, 5}, v2 = {2, 4, 6}, result;
std::merge(v1.begin(), v1.end(), v2.begin(), v2.end(), std::back_inserter(result)); 
// result = {1, 2, 3, 4, 5, 6}

2.23 std::partition, std::stable_partition, std::partition_point

  • std::partition: 将满足条件的元素移动到前端
template<class ForwardIt, class Predicate>
ForwardIt partition(ForwardIt first, ForwardIt last, Predicate pred);

// 示例:按奇偶分区
std::vector<int> v = {1, 2, 3, 4, 5};
auto it = std::partition(v.begin(), v.end(), [](int x) { return x % 2 != 0; });
// 奇数在前,偶数在后(可能改变相对顺序)
  • std::partition_point: 返回分区点
template<class ForwardIt, class Predicate>
ForwardIt partition_point(ForwardIt first, ForwardIt last, Predicate pred);

// 示例:找到分区点
auto it = std::partition_point(v.begin(), v.end(), [](int x) { return x % 2 != 0; });

2.24 std::minelement, std::maxelement, std::clamp

template<class ForwardIt>
ForwardIt min_element(ForwardIt first, ForwardIt last);

// 示例:找到最大值
auto it = std::max_element(v.begin(), v.end());
  • std::clamp: 将元素限制在范围内
template<class T>
const T& clamp(const T& value, const T& lo, const T& hi);

// 示例:限制数值在[0, 100]
int x = 150;
x = std::clamp(x, 0, 100); // x变为100

2.25 std::sample

  • 随机采样,需要插入迭代器
template<class PopulationIt, class SampleIt, class Distance, class UniformRandomBitGenerator>
SampleIt sample(PopulationIt first, PopulationIt last, SampleIt out, Distance n, UniformRandomBitGenerator&& g);

// 示例:随机采样3个元素
std::vector<int> src = {1, 2, 3, 4, 5}, dest;
std::sample(src.begin(), src.end(), std::back_inserter(dest), 3, std::mt19937{});