当前位置: 首页 > news >正文

C++17 中的 std::reduce:详细教程

根据比例生成图片 (4).png

文章目录

    • 1. 简介
    • 2. 函数签名
    • 3. 使用场景
      • 3.1 简单的累加操作
      • 3.2 自定义归并操作
      • 3.3 并行计算的性能优势
    • 4. 注意事项
      • 4.1 归并操作的结合律和交换律
      • 4.2 默认值的使用
    • 5. 总结

1. 简介

std::reduce 是 C++17 标准库中引入的一个算法,用于对范围内的元素进行归并操作。它类似于 std::accumulate,但在某些情况下提供了更灵活的处理方式,尤其是在并行计算方面。

2. 函数签名

std::reduce 的基本函数签名如下:

template<class InputIt>
typename std::iterator_traits<InputIt>::value_type
reduce(InputIt first, InputIt last);

template<class InputIt, class T>
T reduce(InputIt first, InputIt last, T init);

template<class InputIt, class T, class BinaryOperation>
T reduce(InputIt first, InputIt last, T init, BinaryOperation binary_op);

template<class ExecutionPolicy, class ForwardIt>
typename std::iterator_traits<ForwardIt>::value_type
reduce(ExecutionPolicy&& policy, ForwardIt first, ForwardIt last);

template<class ExecutionPolicy, class ForwardIt, class T>
T reduce(ExecutionPolicy&& policy, ForwardIt first, ForwardIt last, T init);

template<class ExecutionPolicy, class ForwardIt, class T, class BinaryOperation>
T reduce(ExecutionPolicy&& policy, ForwardIt first, ForwardIt last, T init, BinaryOperation binary_op);
  • InputIt/ForwardIt:迭代器类型,表示要归并的范围。
  • T:归并操作的初始值类型。
  • BinaryOperation:用于归并的二元操作函数。
  • ExecutionPolicy:执行策略,可以是 std::execution::seq(顺序执行)、std::execution::par(并行执行)或 std::execution::unseq(无序执行)。

3. 使用场景

3.1 简单的累加操作

以下是一个简单的累加示例:

#include <iostream>
#include <vector>
#include <numeric>

int main() {
    std::vector<int> nums = {1, 2, 3, 4, 5};
    int sum = std::reduce(nums.begin(), nums.end());
    std::cout << "Sum: " << sum << std::endl; // 输出 15
    return 0;
}

这里,std::reduce 默认使用加法操作。

3.2 自定义归并操作

可以通过传递自定义的二元操作函数来实现不同的归并逻辑。例如,计算数组中元素的最大值:

#include <iostream>
#include <vector>
#include <numeric>
#include <execution>

int main() {
    std::vector<int> nums = {1, 2, 3, 4, 5};
    int maxElement = std::reduce(std::execution::par, nums.begin(), nums.end(), nums[0], [](int a, int b) {
        return std::max(a, b);
    });
    std::cout << "Maximum element: " << maxElement << std::endl; // 输出 5
    return 0;
}

这里,我们使用了 std::execution::par 来启用并行执行。

3.3 并行计算的性能优势

std::reduce 支持并行执行策略,这使得它在处理大规模数据时能够显著提高性能。例如,计算一个大数组的和:

#include <iostream>
#include <vector>
#include <numeric>
#include <execution>
#include <chrono>

int main() {
    std::vector<int> nums(10000000, 1); // 一个包含 1000 万个元素的数组
    auto start = std::chrono::high_resolution_clock::now();
    int sum = std::reduce(std::execution::par, nums.begin(), nums.end());
    auto end = std::chrono::high_resolution_clock::now();
    auto duration = std::chrono::duration_cast<std::chrono::milliseconds>(end - start).count();
    std::cout << "Sum: " << sum << ", Time taken: " << duration << " ms" << std::endl;
    return 0;
}

通过使用 std::execution::par,std::reduce 可以利用多核处理器进行并行计算。

4. 注意事项

4.1 归并操作的结合律和交换律

std::reduce 的归并操作要求是结合律(Associative)和交换律(Commutative)的。如果归并操作不满足这些性质,结果可能是不确定的。例如,减法操作不满足结合律和交换律,因此在并行执行时可能会导致不同的结果:

std::vector<int> nums = {32, 16, 8, 4, 2, 1};
int result1 = std::reduce(nums.begin() + 1, nums.end(), nums[0], std::minus<>{});
int result2 = std::reduce(std::execution::par, nums.begin() + 1, nums.end(), nums[0], std::minus<>{});
std::cout << result1 << ", " << result2 << std::endl; // 输出可能不同

4.2 默认值的使用

std::reduce 的默认值是元素类型的默认构造值。如果默认值不是归并操作的单位元(Identity Element),可能会导致意外的结果。例如,对于整数类型,加法的单位元是 0,乘法的单位元是 1。

5. 总结

std::reduce 是一个强大且灵活的算法,适用于各种归并操作,尤其是需要并行处理的场景。它与 std::accumulate 类似,但在并行执行方面提供了更好的支持。通过合理使用 std::reduce,可以简化代码并提高性能。

希望这篇教程对你有所帮助!如果有任何问题,欢迎随时提问。

相关文章:

  • foobar2000设置DSP使用教程及软件推荐
  • 成本哪个更低更好用?分析对比大模型OCR、传统OCR和深度学习OCR
  • 28 在可以控制 postgres 服务器, 不知道任何用户名的情况下怎 进入 postgres 服务器
  • 基于springboot轨道交通安全评估系统(源码+lw+部署文档+讲解),源码可白嫖!
  • Windows环境管理多个node版本
  • 2024最新版JavaScript逆向爬虫教程-------基础篇之Chrome开发者工具学习
  • Webpack代码分割、分割策略性能优化详解
  • 记一次Self XSS+CSRF组合利用
  • 第二章:Matlab 编程基础
  • 【C++】class类(类的定义、成员访问、类的大小、this指针)
  • AI 数字人,超越虚拟的实体
  • 【网络安全 | 漏洞挖掘】跨子域账户合并导致的账户劫持与删除
  • Docker安装分布式vLLM
  • java分组接口及校验注解用法
  • 查看notebook的jupyter token
  • 说说高级java每日一道面试题-2025年2月10日-AOP篇-Spring AOP和AspectJ,AOP有什么区别?
  • 信息安全工程师-快速记忆GB17859中的五个安全保护等级
  • 【推荐】碰一碰发视频源码搭建,支持OEM
  • Oracle 19C Database Data Guard 一主两备 -- 生产级
  • Trader Joe‘s EDI 需求分析
  • 盐城市住房城乡建设网站/seo编辑招聘
  • 青岛建网站/一个新品牌如何推广
  • 网站制作哪家公司好/百度竞价排名模式
  • 自己做视频网站用cdn那个便宜/seo快速排名软件平台
  • 宝塔里面一个服务器做多个网站/培训机构排名前十
  • 做视频网站需要什么软件/seo软件服务