C++ accumulate算法 累加与自定义操作

std::accumulate 是 C++ 标准库中的累积算法,通过初始值和二元操作将容器元素归约为单一结果,支持自定义操作如乘积、字符串拼接、最大值查找及复杂对象处理,适用于函数式风格的聚合计算,但不适用于需副作用或提前退出的循环场景。

c++ accumulate算法 累加与自定义操作

C++的

accumulate

算法,说白了,就是个能帮你把容器里一堆东西“揉”成一个结果的工具。它最常见的用法是累加求和,但它的本事远不止于此,通过自定义操作,你能让它干很多意想不到的活儿。我个人觉得,它就像一个多功能搅拌机,你放进去什么,再给它一个搅拌的规则,它就能给你变出你想要的东西。

解决方案

std::accumulate

是 C++ 标准库


头文件中的一个算法,它能够对指定范围内的元素进行累积操作。它有两个主要的重载形式:

基本累加形式

accumulate(first, last, init)

这个版本会将

init

作为初始值,然后依次将范围

[first, last)

中的每个元素加到累积值上。它默认使用加法操作符

+

#include #include #include  // For std::accumulateint main() {    std::vector numbers = {1, 2, 3, 4, 5};    // 初始值为0,对vector中的所有元素进行累加    int sum = std::accumulate(numbers.begin(), numbers.end(), 0);    std::cout << "Sum: " << sum << std::endl; // 输出:Sum: 15    // 初始值为10,对vector中的所有元素进行累加    int sum_from_ten = std::accumulate(numbers.begin(), numbers.end(), 10);    std::cout << "Sum from ten: " << sum_from_ten << std::endl; // 输出:Sum from ten: 25    return 0;}

自定义操作形式

accumulate(first, last, init, binary_op)

这个版本同样以

init

为初始值,但它允许你提供一个二元操作(

binary_op

),这个操作会接受当前的累积值和当前元素作为参数,并返回新的累积值。这才是

accumulate

真正强大之处。

#include #include #include  // For std::accumulate#include   // For string concatenation#include  // For std::multipliesint main() {    std::vector numbers = {1, 2, 3, 4, 5};    // 自定义乘法操作,计算乘积,初始值为1    long long product = std::accumulate(numbers.begin(), numbers.end(), 1LL, std::multiplies());    std::cout << "Product: " << product << std::endl; // 输出:Product: 120 (1*2*3*4*5)    std::vector words = {"Hello", ", ", "world", "!"};    // 自定义字符串拼接操作,初始值为空字符串    std::string sentence = std::accumulate(words.begin(), words.end(), std::string(""),                                           [](const std::string& current_sum, const std::string& element) {                                               return current_sum + element;                                           });    std::cout << "Sentence: " << sentence << std::endl; // 输出:Sentence: Hello, world!    // 也可以使用lambda表达式实现更复杂的逻辑,例如统计偶数个数    int even_count = std::accumulate(numbers.begin(), numbers.end(), 0,                                     [](int count, int num) {                                         return count + (num % 2 == 0 ? 1 : 0);                                     });    std::cout << "Even count: " << even_count << std::endl; // 输出:Even count: 2    return 0;}

C++

accumulate

算法如何实现自定义操作?

实现

accumulate

的自定义操作,关键在于理解并提供一个符合其要求的二元操作符(binary operation)。这个操作符可以是函数对象(functor)、函数指针,或者在现代 C++ 中最常用的 Lambda 表达式。它的签名通常是

ResultType operation(AccumulatedValueType current_sum, ElementType current_element)

当你提供一个自定义的

binary_op

时,

accumulate

的内部逻辑大致是这样的:它会从你给定的

init

值开始,然后遍历容器中的每一个元素。对于每个元素,它会调用

binary_op(当前累积值, 当前元素)

,并将这个调用的结果作为新的累积值。这个过程会一直重复,直到遍历完所有元素,最终返回那个累积值。

立即学习“C++免费学习笔记(深入)”;

举个例子,假设我们想计算一个

std::vector

中所有元素的平均值。虽然直接用

accumulate

求和再除以数量更直观,但我们也可以“硬核”地用

accumulate

来实现一个累积和计数的复合操作。不过,更典型的自定义操作会更直接,比如找出最大值。

#include #include #include #include  // For std::maxint main() {    std::vector data = {3.14, 1.618, 2.718, 0.577};    // 找出vector中的最大值    // 初始值可以设置为一个足够小的值,或者直接用第一个元素    // 这里我们用lambda表达式来做比较操作    double max_val = std::accumulate(data.begin() + 1, data.end(), data[0],                                     [](double current_max, double element) {                                         return std::max(current_max, element);                                     });    std::cout << "Max value: " << max_val << std::endl; // 输出:Max value: 3.14    // 也可以计算所有元素的平方和    double sum_of_squares = std::accumulate(data.begin(), data.end(), 0.0,                                            [](double current_sum, double element) {                                                return current_sum + (element * element);                                            });    std::cout << "Sum of squares: " << sum_of_squares << std::endl; // 输出:Sum of squares: 21.0664    return 0;}

这里,Lambda 表达式

[](double current_max, double element){ return std::max(current_max, element); }

就是我们自定义的二元操作。它接收当前的累积最大值

current_max

和当前元素

element

,然后返回两者中较大的那个。这个例子展示了

accumulate

不仅仅是做简单的加法,它能做任何符合二元操作模式的“聚合”任务。

accumulate

与传统循环或更现代的算法相比,优势与局限性何在?

当我第一次接触

accumulate

的自定义操作时,心里就嘀咕,这不就是个带回调的循环吗?但用着用着,我开始体会到它的一些微妙之处。

优势:

简洁性和表达力: 对于累积操作,尤其是那些可以被归结为“前一个结果与当前元素”的关系的,

accumulate

写出来代码非常紧凑,而且意图清晰。它直接告诉你,我正在“累积”什么。相比于手动写

for

循环,它减少了迭代器管理、初始化和最终返回结果的样板代码。比如,一个简单的求和,用

accumulate

远比

for (int x : vec) sum += x;

来得更函数式,也更不易出错(比如忘记初始化

sum

)。函数式编程风格:

accumulate

鼓励你以函数式的方式思考问题:输入一个范围,一个初始值,一个操作,然后得到一个结果。这有助于编写更纯粹、副作用更少的代码,特别是在现代 C++ 中,这种风格越来越受到推崇。可读性: 对于熟悉标准库算法的开发者来说,看到

std::accumulate

就能立刻明白这段代码是在对一个范围进行归约操作,而不是一个普通的遍历。潜在的优化: 理论上,标准库的实现者可以对

accumulate

进行优化,例如在某些情况下(如

std::reduce

,它是

accumulate

的并行版本)进行并行化。虽然

std::accumulate

本身不是并行算法,但其接口设计为后续的并行算法提供了基础。

局限性:

不适用于所有循环场景:

accumulate

的核心是“归约”成一个单一值。如果你的循环需要执行副作用(例如打印、修改容器、网络请求),或者需要提前退出循环,那么

accumulate

就不适合了。它不是一个通用的循环替代品。初始值的选择: 初始值

init

的类型和值至关重要。如果选择不当,可能会导致结果错误或者类型不匹配。比如计算乘积,初始值必须是

1

而不是

0

。对于某些复杂类型,构造一个合适的初始值可能需要一些思考。可读性下降的风险: 虽然对于简单的累加操作可读性很高,但如果自定义的

binary_op

逻辑过于复杂,或者 Lambda 表达式内部嵌套了太多逻辑,反而会降低代码的可读性,甚至不如一个结构清晰的

for

循环。我见过一些代码,为了“炫技”而强行用

accumulate

实现复杂逻辑,结果维护起来简直是噩梦。性能考量: 对于极度性能敏感的场景,尤其是在旧编译器或特定平台上,手动优化的

for

循环有时可能略快于

accumulate

,因为

accumulate

可能会引入一些函数调用开销。然而,现代编译器通常能很好地优化标准库算法,这种差异往往微乎其微。更重要的是,对于大规模数据并行归约,应该考虑

std::reduce

std::transform_reduce

总的来说,

accumulate

是一个非常棒的工具,尤其是在你需要将一系列元素“折叠”成一个结果时。但就像任何工具一样,理解它的适用场景和局限性,才能真正发挥它的威力,而不是盲目地滥用它。我个人觉得,当你发现自己写了一个

for

循环,它的主要目的是计算一个总和、一个乘积、一个最大/最小值,或者将一系列元素拼接起来时,就应该考虑

accumulate

了。

accumulate

在处理不同数据类型和复杂对象时的表现如何?

accumulate

在处理不同数据类型和复杂对象时,其核心能力并没有改变,依然是基于你提供的二元操作来“揉”数据。它的灵活性主要体现在

binary_op

的设计上。

基本数据类型:对于

int

,

double

,

float

等基本数值类型,

accumulate

表现得非常自然,无论是累加、累乘,还是其他简单的数值操作,都非常直观。类型推导通常也没什么问题,只要初始值类型设置得当。

#include #include #include int main() {    std::vector prices = {19.99, 29.50, 5.00, 12.75};    double total_cost = std::accumulate(prices.begin(), prices.end(), 0.0);    std::cout << "Total cost: " << total_cost << std::endl; // 输出:Total cost: 67.24    return 0;}

字符串类型:字符串拼接是

accumulate

处理复杂对象的一个经典例子。由于

std::string

支持

+

操作符进行拼接,所以

accumulate

可以很方便地实现字符串的连接。

#include #include #include #include int main() {    std::vector parts = {"The ", "quick ", "brown ", "fox."};    std::string full_sentence = std::accumulate(parts.begin(), parts.end(), std::string(""));    std::cout << "Full sentence: " << full_sentence << std::endl; // 输出:Full sentence: The quick brown fox.    return 0;}

这里需要注意的是,如果

init

是一个 C 风格字符串字面量(例如

""

),它会被推断为

const char*

,导致后续的

+

操作符行为不符合预期。所以,明确地用

std::string("")

来初始化非常重要。

自定义类或结构体:这是

accumulate

真正展现其灵活性的地方。只要你的自定义类支持你想要执行的二元操作,或者你可以提供一个 Lambda 表达式/函数对象来定义这个操作,

accumulate

就能工作。

假设我们有一个

Product

结构体,我们想计算所有产品的总库存价值。

#include #include #include #include struct Product {    std::string name;    double price;    int quantity;    double get_value() const {        return price * quantity;    }};int main() {    std::vector inventory = {        {"Laptop", 1200.0, 5},        {"Mouse", 25.0, 50},        {"Keyboard", 75.0, 20}    };    // 计算总库存价值    // 初始值是0.0,累加的是每个产品的价值    double total_inventory_value = std::accumulate(inventory.begin(), inventory.end(), 0.0,                                                   [](double current_total, const Product& p) {                                                       return current_total + p.get_value();                                                   });    std::cout << "Total inventory value: " << total_inventory_value << std::endl; // 输出:Total inventory value: 7650    return 0;}

在这个例子中,

accumulate

遍历

Product

对象,但我们自定义的 Lambda 表达式负责从每个

Product

对象中提取

get_value()

并将其加到累积总和中。这完美展示了

accumulate

能够处理复杂对象的“内部”数据,并将其归约为一个简单的数值。

需要注意的细节:

初始值类型:

accumulate

的结果类型是由

init

参数的类型决定的。如果你用

0

作为初始值来累加

double

类型的数据,结果可能是

int

,导致精度丢失。所以,对于

double

累加,初始值应为

0.0

操作符重载: 如果你的自定义类需要进行默认的加法操作,你可以重载

operator+

。但通常情况下,使用 Lambda 表达式或函数对象来定义特定的聚合逻辑更为灵活,也避免了对类本身的侵入性修改。性能: 对于复杂对象的累积,性能瓶颈通常不在

accumulate

算法本身,而在于你

binary_op

内部执行的操作。如果

binary_op

涉及大量计算或资源分配,那么这部分会是主要的性能开销。

总而言之,

accumulate

是一个非常通用的算法,它的“魔力”在于能够让你完全掌控累积过程中的每一步。只要你能用一个二元操作来描述如何将“当前累积结果”和“下一个元素”结合起来,

accumulate

就能胜任。这使得它在处理各种数据类型和复杂对象时都游刃有余。

以上就是C++ accumulate算法 累加与自定义操作的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1473289.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月18日 20:19:31
下一篇 2025年12月18日 20:19:40

相关推荐

  • Uniapp 中如何不拉伸不裁剪地展示图片?

    灵活展示图片:如何不拉伸不裁剪 在界面设计中,常常需要以原尺寸展示用户上传的图片。本文将介绍一种在 uniapp 框架中实现该功能的简单方法。 对于不同尺寸的图片,可以采用以下处理方式: 极端宽高比:撑满屏幕宽度或高度,再等比缩放居中。非极端宽高比:居中显示,若能撑满则撑满。 然而,如果需要不拉伸不…

    2025年12月24日
    400
  • 如何让小说网站控制台显示乱码,同时网页内容正常显示?

    如何在不影响用户界面的情况下实现控制台乱码? 当在小说网站上下载小说时,大家可能会遇到一个问题:网站上的文本在网页内正常显示,但是在控制台中却是乱码。如何实现此类操作,从而在不影响用户界面(UI)的情况下保持控制台乱码呢? 答案在于使用自定义字体。网站可以通过在服务器端配置自定义字体,并通过在客户端…

    2025年12月24日
    800
  • 如何在地图上轻松创建气泡信息框?

    地图上气泡信息框的巧妙生成 地图上气泡信息框是一种常用的交互功能,它简便易用,能够为用户提供额外信息。本文将探讨如何借助地图库的功能轻松创建这一功能。 利用地图库的原生功能 大多数地图库,如高德地图,都提供了现成的信息窗体和右键菜单功能。这些功能可以通过以下途径实现: 高德地图 JS API 参考文…

    2025年12月24日
    400
  • 如何使用 scroll-behavior 属性实现元素scrollLeft变化时的平滑动画?

    如何实现元素scrollleft变化时的平滑动画效果? 在许多网页应用中,滚动容器的水平滚动条(scrollleft)需要频繁使用。为了让滚动动作更加自然,你希望给scrollleft的变化添加动画效果。 解决方案:scroll-behavior 属性 要实现scrollleft变化时的平滑动画效果…

    2025年12月24日
    000
  • 如何为滚动元素添加平滑过渡,使滚动条滑动时更自然流畅?

    给滚动元素平滑过渡 如何在滚动条属性(scrollleft)发生改变时为元素添加平滑的过渡效果? 解决方案:scroll-behavior 属性 为滚动容器设置 scroll-behavior 属性可以实现平滑滚动。 html 代码: click the button to slide right!…

    2025年12月24日
    500
  • 如何选择元素个数不固定的指定类名子元素?

    灵活选择元素个数不固定的指定类名子元素 在网页布局中,有时需要选择特定类名的子元素,但这些元素的数量并不固定。例如,下面这段 html 代码中,activebar 和 item 元素的数量均不固定: *n *n 如果需要选择第一个 item元素,可以使用 css 选择器 :nth-child()。该…

    2025年12月24日
    200
  • 使用 SVG 如何实现自定义宽度、间距和半径的虚线边框?

    使用 svg 实现自定义虚线边框 如何实现一个具有自定义宽度、间距和半径的虚线边框是一个常见的前端开发问题。传统的解决方案通常涉及使用 border-image 引入切片图片,但是这种方法存在引入外部资源、性能低下的缺点。 为了避免上述问题,可以使用 svg(可缩放矢量图形)来创建纯代码实现。一种方…

    2025年12月24日
    100
  • 旋转长方形后,如何计算其相对于画布左上角的轴距?

    绘制长方形并旋转,计算旋转后轴距 在拥有 1920×1080 画布中,放置一个宽高为 200×20 的长方形,其坐标位于 (100, 100)。当以任意角度旋转长方形时,如何计算它相对于画布左上角的 x、y 轴距? 以下代码提供了一个计算旋转后长方形轴距的解决方案: const x = 200;co…

    2025年12月24日
    000
  • 旋转长方形后,如何计算它与画布左上角的xy轴距?

    旋转后长方形在画布上的xy轴距计算 在画布中添加一个长方形,并将其旋转任意角度,如何计算旋转后的长方形与画布左上角之间的xy轴距? 问题分解: 要计算旋转后长方形的xy轴距,需要考虑旋转对长方形宽高和位置的影响。首先,旋转会改变长方形的长和宽,其次,旋转会改变长方形的中心点位置。 求解方法: 计算旋…

    2025年12月24日
    000
  • 旋转长方形后如何计算其在画布上的轴距?

    旋转长方形后计算轴距 假设长方形的宽、高分别为 200 和 20,初始坐标为 (100, 100),我们将它旋转一个任意角度。根据旋转矩阵公式,旋转后的新坐标 (x’, y’) 可以通过以下公式计算: x’ = x * cos(θ) – y * sin(θ)y’ = x * …

    2025年12月24日
    000
  • 如何让“元素跟随文本高度,而不是撑高父容器?

    如何让 元素跟随文本高度,而不是撑高父容器 在页面布局中,经常遇到父容器高度被子元素撑开的问题。在图例所示的案例中,父容器被较高的图片撑开,而文本的高度没有被考虑。本问答将提供纯css解决方案,让图片跟随文本高度,确保父容器的高度不会被图片影响。 解决方法 为了解决这个问题,需要将图片从文档流中脱离…

    2025年12月24日
    000
  • 如何计算旋转后长方形在画布上的轴距?

    旋转后长方形与画布轴距计算 在给定的画布中,有一个长方形,在随机旋转一定角度后,如何计算其在画布上的轴距,即距离左上角的距离? 以下提供一种计算长方形相对于画布左上角的新轴距的方法: const x = 200; // 初始 x 坐标const y = 90; // 初始 y 坐标const w =…

    2025年12月24日
    200
  • CSS元素设置em和transition后,为何载入页面无放大效果?

    css元素设置em和transition后,为何载入无放大效果 很多开发者在设置了em和transition后,却发现元素载入页面时无放大效果。本文将解答这一问题。 原问题:在视频演示中,将元素设置如下,载入页面会有放大效果。然而,在个人尝试中,并未出现该效果。这是由于macos和windows系统…

    2025年12月24日
    200
  • 为什么 CSS mask 属性未请求指定图片?

    解决 css mask 属性未请求图片的问题 在使用 css mask 属性时,指定了图片地址,但网络面板显示未请求获取该图片,这可能是由于浏览器兼容性问题造成的。 问题 如下代码所示: 立即学习“前端免费学习笔记(深入)”; icon [data-icon=”cloud”] { –icon-cl…

    2025年12月24日
    200
  • 如何利用 CSS 选中激活标签并影响相邻元素的样式?

    如何利用 css 选中激活标签并影响相邻元素? 为了实现激活标签影响相邻元素的样式需求,可以通过 :has 选择器来实现。以下是如何具体操作: 对于激活标签相邻后的元素,可以在 css 中使用以下代码进行设置: li:has(+li.active) { border-radius: 0 0 10px…

    2025年12月24日
    100
  • 如何模拟Windows 10 设置界面中的鼠标悬浮放大效果?

    win10设置界面的鼠标移动显示周边的样式(探照灯效果)的实现方式 在windows设置界面的鼠标悬浮效果中,光标周围会显示一个放大区域。在前端开发中,可以通过多种方式实现类似的效果。 使用css 使用css的transform和box-shadow属性。通过将transform: scale(1.…

    2025年12月24日
    200
  • 如何计算旋转后的长方形在画布上的 XY 轴距?

    旋转长方形后计算其画布xy轴距 在创建的画布上添加了一个长方形,并提供其宽、高和初始坐标。为了视觉化旋转效果,还提供了一些旋转特定角度后的图片。 问题是如何计算任意角度旋转后,这个长方形的xy轴距。这涉及到使用三角学来计算旋转后的坐标。 以下是一个 javascript 代码示例,用于计算旋转后长方…

    2025年12月24日
    000
  • 为什么我的 Safari 自定义样式表在百度页面上失效了?

    为什么在 Safari 中自定义样式表未能正常工作? 在 Safari 的偏好设置中设置自定义样式表后,您对其进行测试却发现效果不同。在您自己的网页中,样式有效,而在百度页面中却失效。 造成这种情况的原因是,第一个访问的项目使用了文件协议,可以访问本地目录中的图片文件。而第二个访问的百度使用了 ht…

    2025年12月24日
    000
  • 如何用前端实现 Windows 10 设置界面的鼠标移动探照灯效果?

    如何在前端实现 Windows 10 设置界面中的鼠标移动探照灯效果 想要在前端开发中实现 Windows 10 设置界面中类似的鼠标移动探照灯效果,可以通过以下途径: CSS 解决方案 DEMO 1: Windows 10 网格悬停效果:https://codepen.io/tr4553r7/pe…

    2025年12月24日
    000
  • 使用CSS mask属性指定图片URL时,为什么浏览器无法加载图片?

    css mask属性未能加载图片的解决方法 使用css mask属性指定图片url时,如示例中所示: mask: url(“https://api.iconify.design/mdi:apple-icloud.svg”) center / contain no-repeat; 但是,在网络面板中却…

    2025年12月24日
    000

发表回复

登录后才能评论
关注微信