MongoDB Aggregation:使用 $sum 正确计算数组中整数的总和

mongodb aggregation:使用 $sum 正确计算数组中整数的总和

本文介绍了如何使用 MongoDB 的聚合管道和 $sum 操作符来正确计算嵌套在数组中的整数的总和。重点讲解了在使用 $sum 时可能遇到的问题,并提供了使用 $map 和 $reduce 操作符的两种解决方案,并附带示例代码和 MongoDB Playground 链接,方便读者理解和实践。

在使用 MongoDB 的聚合管道时,经常需要计算数组中元素的总和。$sum 操作符是实现这一功能的关键。然而,当数组嵌套在文档的深层结构中时,直接使用 $sum 可能会导致意外的结果,例如返回 0 而不是实际的总和。本文将深入探讨这个问题,并提供两种有效的解决方案。

问题分析

假设我们有如下结构的文档:

{  "_id": "2023-04-08",  "dailyNotes": [    {      "therapyDiscipline": "PT",      "individual": [        60,        45,        30,        75,        55      ],      "concurrent": [        69      ],      "coTreat": [        67      ]    },    {      "therapyDiscipline": "YT",      "individual": [        2,        4      ]    }  ]}

我们的目标是计算 dailyNotes 数组中每个元素的 individual 数组的总和,并将其作为新的 individual 字段添加到 dailyNotes 数组的每个元素中。

如果直接使用 $sum 操作符,例如:

{  $group: {    _id: '$_id',    individual: {      $sum: '$dailyNotes.individual'    }  }}

或者:

{  $project: {    individual: {      $sum: '$dailyNotes.individual'    }  }}

可能会错误地返回 0。这是因为 $sum 操作符直接应用于 dailyNotes.individual 时,MongoDB 无法正确地遍历嵌套数组。

解决方案一:使用 $map

$map 操作符允许我们遍历数组中的每个元素,并对每个元素应用一个表达式。我们可以利用 $map 来遍历 dailyNotes 数组,并对每个元素的 individual 数组使用 $sum 操作符。

db.collection.aggregate([  {    $project: {      _id: 1,      dailyNotes: {        $map: {          input: "$dailyNotes",          in: { individual: { $sum: "$$this.individual" } }        }      }    }  }])

在这个聚合管道中:

$project 阶段用于选择需要保留的字段,并使用 $map 创建新的 dailyNotes 数组。$map 操作符遍历 dailyNotes 数组。input: “$dailyNotes” 指定要遍历的数组。in: { individual: { $sum: “$$this.individual” } } 定义了应用于每个元素的表达式。$$this 代表当前正在遍历的元素,$$this.individual 则指向当前元素的 individual 数组。$sum: “$$this.individual” 计算 individual 数组的总和,并将结果赋值给新的 individual 字段。

MongoDB Playground 示例

解决方案二:使用 $reduce

$reduce 操作符也允许我们遍历数组,并将其归约为单个值。我们可以利用 $reduce 来遍历 dailyNotes 数组,并将每个元素的 individual 数组的总和添加到结果数组中。

db.collection.aggregate([  {    $project: {      _id: 1,      dailyNotes: {        $reduce: {          input: "$dailyNotes",          initialValue: [],          in: { $concatArrays: [ "$$value", [ { individual: { $sum: "$$this.individual" } } ] ] }        }      }    }  }])

在这个聚合管道中:

$project 阶段用于选择需要保留的字段,并使用 $reduce 创建新的 dailyNotes 数组。$reduce 操作符遍历 dailyNotes 数组。input: “$dailyNotes” 指定要遍历的数组。initialValue: [] 定义了初始值,即一个空数组。in: { $concatArrays: [ “$$value”, [ { individual: { $sum: “$$this.individual” } } ] ] } 定义了应用于每个元素的表达式。$$value 代表当前的结果数组,$$this 代表当前正在遍历的元素。$sum: “$$this.individual” 计算 individual 数组的总和,并将结果封装在一个新的对象 { individual: … } 中。$concatArrays 将新的对象添加到结果数组中。

MongoDB Playground 示例

总结

当需要计算嵌套在数组中的整数的总和时,直接使用 $sum 操作符可能无法得到正确的结果。使用 $map 或 $reduce 操作符可以有效地解决这个问题。$map 适用于直接转换数组中的每个元素,而 $reduce 适用于将数组归约为单个值。选择哪种方案取决于具体的需求和数据结构。在实际应用中,建议根据数据量和性能要求选择合适的方案。

以上就是MongoDB Aggregation:使用 $sum 正确计算数组中整数的总和的详细内容,更多请关注创想鸟其它相关文章!

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

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月20日 05:39:20
下一篇 2025年12月20日 05:39:37

相关推荐

  • JavaScript如何用数组的every方法检测全部

    javascript的every方法用于检测数组所有元素是否都满足指定条件,返回布尔值。1. 若所有元素均通过测试,返回true;若任一元素未通过,则立即返回false并停止遍历。2. 其语法为arr.every(callback[, thisarg]),callback需返回布尔值。3. 与som…

    2025年12月20日 好文分享
    000
  • 如何用BOM获取用户的蓝牙设备信息?

    不能直接通过bom获取用户所有蓝牙设备信息,必须通过web bluetooth api在用户授权后有目的地连接。开发者需使用navigator.bluetooth.requestdevice()方法触发浏览器弹窗,让用户手动选择设备;代码须运行于https环境并由用户手势触发;api设计强制安全上下…

    2025年12月20日 好文分享
    000
  • Promise链式调用的技巧分享

    promise链式调用通过then()方法将多个异步操作串联,使代码更清晰且避免回调地狱。1. 每个then()返回新promise,状态取决于回调返回值;2. 可使用catch()统一捕获链中错误;3. 长链可通过拆分函数或使用async/await提升可读性;4. 并发操作可用promise.a…

    2025年12月20日 好文分享
    000
  • MongoDB聚合管道:使用$sum计算数组元素之和返回0的解决方案

    本文旨在解决MongoDB聚合管道中使用$sum操作符计算嵌套在数组中的数值元素之和时,意外返回0的问题。通过分析问题原因,并提供使用$map和$reduce操作符的两种解决方案,帮助开发者正确计算数组元素的总和。 在MongoDB的聚合管道中,使用$sum操作符计算数组元素的总和是一个常见的需求。…

    2025年12月20日
    000
  • MongoDB聚合操作中 $sum 错误返回0的解决方法

    本文针对MongoDB聚合操作中使用 $sum 运算符时,未能正确计算数组元素总和,而是返回0的问题,提供了详细的解决方案。通过 $map 和 $reduce 运算符,结合实际案例,展示了如何正确地对嵌套在文档中的数组进行求和,并生成所需格式的结果。 在使用MongoDB进行数据聚合时,经常会遇到需…

    2025年12月20日
    000
  • JavaScript的Math.floor方法是什么?如何使用?

    math.floor() 是向下取整函数,返回小于或等于给定数字的最大整数。例如:math.floor(5.95) 返回 5,math.floor(-5.05) 返回 -6。其应用场景包括:1. 分页计算中确定当前页码;2. 数组索引生成,确保索引为整数;3. 游戏开发中将浮点坐标转为整数坐标;4.…

    2025年12月20日 好文分享
    000
  • 使用Promise实现延迟执行

    使用promise实现延迟执行的核心在于将settimeout包装为promise,以支持链式调用和async/await。通过创建一个在settimeout回调中调用resolve的promise,可实现非阻塞的延迟操作;例如:function delay(ms) { return new pro…

    2025年12月20日 好文分享
    000
  • JavaScript如何用Intl.DateTimeFormat格式化日期

    intl.datetimeformat 是 javascript 中用于国际化日期和时间格式化的强大工具。1. 它能根据指定的语言环境自动处理日期格式、月份名称、星期几、时区等;2. 支持通过 options 对象精细控制年、月、日、时、分、秒等组件的显示方式;3. 提供 datestyle 和 t…

    2025年12月20日 好文分享
    000
  • 如何用BOM获取用户的GPS位置信息?

    使用geolocation api而非bom获取用户gps位置信息是正确做法。1. bom无法直接访问硬件或隐私数据,必须通过geolocation api实现;2. 使用前需检查浏览器支持:if (“geolocation” in navigator);3. 获取当前位置用…

    2025年12月20日 好文分享
    000
  • async函数与回调函数的对比

    async函数和回调函数是javascript中处理异步操作的不同方式,其核心区别在于代码结构、可读性和错误处理机制。1. 回调函数通过将函数作为参数传递,在异步操作完成后执行,但容易导致“回调地狱”,代码嵌套深、逻辑混乱;2. async/await基于promise,使用同步风格编写异步代码,提…

    2025年12月20日 好文分享
    000
  • JavaScript中回调函数的使用方法

    回调函数在javascript中本质上是将函数作为参数传递给另一函数,并在特定条件满足时执行,它广泛应用于异步编程。解决方案包括:1. 作为参数传递,如greet函数调用时传入saygoodbye作为回调;2. 处理异步操作,如settimeout中两秒后执行回调;3. 事件处理,如按钮点击触发回调…

    2025年12月20日 好文分享
    000
  • 如何处理多个异步任务并行执行

    javascript中处理多个异步任务并行执行的核心方法包括:1.使用promise.all实现“全有或全无”的并发控制,适用于所有数据必须成功获取才能继续执行的场景;2.使用promise.allsettled确保所有任务无论成功或失败都能完成,便于更细致地处理每个结果;3.通过限制并发数(如实现…

    2025年12月20日 好文分享
    000
  • 使用 JavaScript 实现平滑的角色移动:解决键盘重复延迟问题

    本文旨在解决使用 JavaScript 开发游戏时,通过 WASD 键控制角色移动时出现的“初始移动缓慢,之后快速重复”的问题。我们将探讨如何使用 setInterval 函数来创建一个连续移动的循环,并优化键盘事件处理,从而实现平滑流畅的角色移动效果。 在开发基于键盘控制的 JavaScript …

    2025年12月20日
    000
  • 如何处理异步函数的副作用

    异步函数的副作用源于其非阻塞和时间不确定性,导致状态变化难以预测。1. 利用promise或async/await封装副作用,构建清晰执行链;2. 强化错误处理机制,通过.catch()或try…catch确保异常可控;3. 引入redux、vuex等状态管理工具,实现状态变更可追踪;4…

    2025年12月20日 好文分享
    000
  • JavaScript如何利用解构赋值提取对象属性

    解构赋值解决了属性提取繁琐、代码冗余的问题,提升了可读性和维护性。1. 它允许从对象或数组中直接提取数据并赋值给变量,避免重复书写属性访问代码;2. 支持重命名、设置默认值、嵌套提取、结合剩余操作符、函数参数解构等高级用法;3. 在处理api响应、配置对象时尤为高效;4. 常见注意事项包括:默认值仅…

    2025年12月20日 好文分享
    000
  • JavaScript的String.prototype.split方法是什么?怎么用?

    split()方法根据分隔符将字符串拆分为数组。1. separator参数可为字符串或正则,决定分割位置;2. limit参数限制返回数组长度;3. 使用正则可处理复杂分隔模式,如多个分隔符或捕获组保留分隔符信息;4. 分隔符不存在时返回原字符串数组,为空字符串时逐字符分割;5. 结合map、fi…

    2025年12月20日 好文分享
    000
  • JavaScript中异步流程控制方法

    javascript需要异步流程控制是因为其单线程特性,若执行耗时任务(如网络请求)会阻塞主线程,导致页面卡死。1. 异步通过事件循环机制,将任务交给浏览器其他线程处理,主线程继续执行后续代码;2. 回调函数是最早解决方案,但易形成“回调地狱”;3. promise提供结构化方式,支持链式调用和集中…

    2025年12月20日 好文分享
    000
  • ES6的箭头函数与传统函数有何区别

    箭头函数与传统函数的核心差异在于this绑定、arguments对象、构造函数支持及语法简洁性。1.this绑定:传统函数动态绑定this,取决于调用方式;箭头函数词法绑定this,继承自父级作用域。2.arguments对象:传统函数有arguments对象,箭头函数无,需用剩余参数替代。3.构造…

    2025年12月20日 好文分享
    000
  • JavaScript中异步模块加载机制

    javascript中的异步模块加载机制通过按需非阻塞加载提升网页性能。1. 早期使用标签同步加载导致页面阻塞,全局变量污染和依赖混乱问题严重;2. amd规范以requirejs为代表,通过define()和require()实现异步加载,但语法冗余;3. commonjs用于node.js环境,…

    2025年12月20日 好文分享
    000
  • ES6中如何用import动态加载模块

    动态加载模块的核心目的是为了提升性能和用户体验。它通过减少初始加载体积、优化资源利用、提升用户感知速度、实现更好的缓存策略,尤其适用于大型应用的代码分割和懒加载场景。import()函数与传统import声明的区别在于:1. import()是动态的、可在运行时根据条件调用,而传统import是静态…

    2025年12月20日 好文分享
    000

发表回复

登录后才能评论
关注微信