JavaScript中根据键值比较两个对象并计算总和的教程

JavaScript中根据键值比较两个对象并计算总和的教程

本教程旨在指导开发者如何在javascript中,依据一个嵌套对象(`values`)的键值,从另一个对象(`points`)中匹配并计算相应分数的总和。文章提供了多种实现策略,包括使用`reduce`进行迭代聚合,以及通过构建查找表进行高效数据匹配和求和,以满足复杂的数据处理需求。

前端开发中,我们经常会遇到需要处理结构化数据并进行聚合计算的场景。一个典型的例子是,我们有两个JavaScript对象:一个包含用户选择或状态的嵌套对象,另一个则存储了这些选择对应的分数。我们的目标是根据第一个对象的键值,从第二个对象中查找对应的分数并计算总和。

考虑以下两个JavaScript对象作为示例数据:

const values = {    Q1: {        Q1A1: "Yes",    },    Q2: {        Q2A1: "Yes",    },    Q3: {        Q3A2: "No",    },};const points = {    Q1A1: 41,    Q1A2: 0,    Q2A1: 19,    Q2A2: 0,    Q3A1: 25,    Q3A2: 0, // 假设这里是0,如果题目要求Q3A2="No"不计分,则即使这里是5也应忽略};

我们的目标是根据values对象中嵌套的键(例如Q1A1、Q2A1)以及它们的值(例如”Yes”),从points对象中查找对应的分数并求和。例如,如果Q1A1是”Yes”,则计入points.Q1A1的分数;如果Q3A2是”No”,则不计分。最终结果应为41 + 19 = 60。

接下来,我们将探讨几种实现这一目标的JavaScript方法。

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

方法一:使用嵌套 reduce 进行直接聚合

这种方法通过两次 reduce 迭代,直接遍历 values 对象的结构,并在满足条件时累加分数。

const values = {    Q1: { Q1A1: "Yes" },    Q2: { Q2A1: "Yes" },    Q3: { Q3A2: "No" },};const points = {    Q1A1: 41,    Q1A2: 0,    Q2A1: 19,    Q2A2: 0,    Q3A1: 25,    Q3A2: 5, // 假设这里是5,但由于Q3A2="No",此值不应被计入};const total = Object.values(values) // 获取values对象中所有Q1, Q2, Q3等子对象  .reduce((acc, cur) => { // 外层reduce用于遍历Q1, Q2, Q3等    return acc + Object.entries(cur) // 获取当前子对象(如{Q1A1: "Yes"})的键值对      .reduce((accInner, [key, val]) => { // 内层reduce用于遍历子对象中的键值对        // 判断条件:值不为'No' 且 points对象中存在对应的key,且其值为有效分数(非0或undefined)        // 注意:如果points[key]可能是0,但0也应该计入,则只需判断 points[key] !== undefined        if (val !== 'No' && points[key] !== undefined && points[key] !== null) {          return accInner + points[key];        }        return accInner;      }, 0); // 内层reduce的初始累加值为0  }, 0); // 外层reduce的初始累加值为0console.log(total); // 输出: 60

解析:

Object.values(values):首先提取 values 对象中所有第一层的值,即 Q1, Q2, Q3 等子对象。外层 reduce:遍历这些子对象。acc 是总累加器,cur 是当前的子对象(例如 { Q1A1: “Yes” })。Object.entries(cur):对于每个子对象,提取其键值对数组(例如 [[‘Q1A1’, ‘Yes’]])。内层 reduce:遍历子对象的键值对。accInner 是当前子对象的分数累加器,[key, val] 是当前的键值对。条件判断:val !== ‘No’ 确保只有非”No”的答案才考虑计分。points[key] !== undefined && points[key] !== null 确保在 points 对象中存在对应的键,并且其值有效。如果points[key]可能为0且0也应计入,则只需 points[key] !== undefined。如果条件满足,将 points[key] 的值加到 accInner 中。

这种方法简洁,直接在迭代过程中完成计算,适用于数据结构相对固定的情况。

方法二:构建查找表后进行过滤和求和

当数据量较大或需要更清晰地分离数据处理步骤时,可以先从 values 对象中构建一个符合条件的键的查找表,然后再用这个查找表去过滤和累加 points 对象中的分数。

const values = {    Q1: { Q1A1: "Yes" },    Q2: { Q2A1: "Yes" },    Q3: { Q3A2: "No" },};const points = {    Q1A1: 41,    Q1A2: 0,    Q2A1: 19,    Q2A2: 0,    Q3A1: 25,    Q3A2: 5,};// 步骤1: 从values对象中构建一个符合条件的键的查找表const lookupKeys = Object.values(values).reduce((acc, cur) => {  // 假设每个子对象只有一个键值对  const [key, val] = Object.entries(cur)[0];  if (val === "Yes") { // 只有值为"Yes"的键才被添加到查找表中    acc[key] = true; // 使用布尔值作为查找表的值,表示该键存在且符合条件  }  return acc;}, {});console.log("Lookup Table:", lookupKeys); // 输出: { Q1A1: true, Q2A1: true }// 步骤2: 遍历points对象,根据查找表过滤并求和const total = Object.entries(points)  .filter(([key, score]) => lookupKeys[key] && score !== 0) // 过滤条件:键在查找表中存在 且 分数不为0  .reduce((accumulator, [key, score]) => accumulator + score, 0); // 对过滤后的分数进行累加console.log(total); // 输出: 60

解析:

构建 lookupKeys:Object.values(values).reduce(…):遍历 values 对象的子对象。Object.entries(cur)[0]:获取每个子对象内部的第一个(也是唯一一个)键值对。if (val === “Yes”) acc[key] = true;:如果值是 “Yes”,则将该键添加到 lookupKeys 对象中,值为 true,表示这是一个需要计分的有效键。过滤并求和:Object.entries(points):将 points 对象转换为键值对数组。.filter(([key, score]) => lookupKeys[key] && score !== 0):过滤这些键值对。只有当键存在于 lookupKeys 中(即 lookupKeys[key] 为 true)并且分数不为 0 时,才保留该项。.reduce((accumulator, [key, score]) => accumulator + score, 0):对过滤后的项进行累加,得到最终总和。

这种方法将数据准备和计算分成了两个清晰的阶段,提高了代码的可读性和模块化。

方法三:提取所有相关键到 Set 后进行过滤和求和

如果我们的条件仅仅是“values 对象中存在的键,无论其值是Yes还是No,只要在points中有对应分数就计入”,或者我们只关心特定条件的键(如”Yes”),并希望高效地检查键的存在性,可以使用 Set 结构来存储符合条件的键。

const values = {    Q1: { Q1A1: "Yes" },    Q2: { Q2A1: "Yes" },    Q3: { Q3A2: "No" },};const points = {    Q1A1: 41,    Q1A2: 0,    Q2A1: 19,    Q2A2: 0,    Q3A1: 25,    Q3A2: 5,};// 步骤1: 从values对象中提取所有符合条件的键到Setconst pointKeysToSum = Object.values(values).reduce((acc, cur) => {  const [key, val] = Object.entries(cur)[0]; // 获取子对象中的键值对  if (val === "Yes") { // 假设我们只关心值为"Yes"的键    acc.add(key); // 将符合条件的键添加到Set中  }  return acc;}, new Set()); // 初始值为一个空的Setconsole.log("Keys to Sum (Set):", pointKeysToSum); // 输出: Set(2) { 'Q1A1', 'Q2A1' }// 步骤2: 遍历points对象,根据Set过滤并求和const total = Object.entries(points)  .filter(([key, score]) => pointKeysToSum.has(key)) // 过滤条件:键存在于Set中  .reduce((accumulator, [key, score]) => accumulator + score, 0); // 对过滤后的分数进行累加console.log(total); // 输出: 60

解析:

构建 pointKeysToSum Set:Object.values(values).reduce(…):遍历 values 对象的子对象。if (val === “Yes”) acc.add(key);:根据条件(此处为 val === “Yes”)将键添加到 Set 中。Set 自动处理重复键,确保每个键只出现一次。过滤并求和:Object.entries(points):将 points 对象转换为键值对数组。.filter(([key, score]) => pointKeysToSum.has(key)):使用 Set.prototype.has() 方法高效地检查键是否存在于 pointKeysToSum 中。.reduce((accumulator, [key, score]) => accumulator + score, 0):对过滤后的项进行累加。

Set 的 has() 方法具有平均 O(1) 的时间复杂度,这使得它在处理大量键的查找时比数组的 includes() 方法(O(n))更高效。

注意事项与总结

条件逻辑: 教程中演示了根据 val === “Yes” 或 val !== “No” 进行判断。实际应用中,你需要根据具体业务需求调整条件。键值类型: 确保 values 中提取的键与 points 中的键类型一致(通常都是字符串)。分数有效性: 在累加分数时,考虑 points[key] 可能为 undefined、null 或 0 的情况。根据业务需求决定这些值是否应该计入总和。数据结构假设: 示例中 values 对象的每个子对象(如 Q1)内部只有一个键值对。如果存在多个,你需要调整 Object.entries(cur)[0] 的逻辑,可能需要再次迭代。性能考量:对于小型数据集,所有方法性能差异不大。对于大型数据集,Set 或构建查找表(方法二和方法三)通常比嵌套 reduce(方法一)更具优势,因为它们将查找操作的复杂度从 O(N*M) 降低到 O(N+M),其中 N 和 M 分别是两个对象的大小。Set.prototype.has() 的 O(1) 平均时间复杂度在查找效率上表现突出。可读性: 选择最能清晰表达意图的方法。对于复杂逻辑,分阶段构建查找表或 Set 可能比深度嵌套的 reduce 更易读。

通过本文介绍的几种方法,你可以根据实际需求和偏好,灵活选择在JavaScript中根据键值比较两个对象并计算总和的最佳策略。理解这些数组和对象方法的强大功能,将有助于你更高效地处理各种数据聚合任务。

以上就是JavaScript中根据键值比较两个对象并计算总和的教程的详细内容,更多请关注创想鸟其它相关文章!

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

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月21日 13:35:51
下一篇 2025年12月21日 13:36:03

相关推荐

  • Javascript如何进行模块化开发?

    JavaScript模块化开发是将代码拆分为独立、可复用、有明确依赖关系的文件,核心方式为ES Module(推荐)和CommonJS;ESM语法简洁、支持tree-shaking,需type=”module”或.mjs;CommonJS适用于旧Node.js项目;实际项目需…

    2025年12月21日
    000
  • 如何实现定时触发与自动关闭弹出窗口的联动机制

    本文详细探讨了在特定CMS环境中,如何通过JavaScript的`setInterval`函数实现定时触发一个功能(例如强制刷新视频缩略图),并紧接着自动关闭触发该功能的弹出窗口。核心在于利用两个错开的`setInterval`调用,一个用于打开弹出,另一个稍后用于关闭,从而在不影响用户体验的前提下…

    2025年12月21日
    000
  • 解决页面刷新后暗模式切换图标不同步的问题

    本教程详细讲解如何确保网页的暗模式切换图标在页面刷新后依然能正确反映当前的暗模式状态。通过分析原始代码的问题,我们将展示如何利用 `localstorage` 存储的状态,在页面加载时同步更新图标的显示,从而提供一致的用户体验。核心在于修改切换函数以同时管理图标可见性,并在页面初始化时根据存储状态调…

    2025年12月21日
    000
  • javascript的尾调用优化是什么_它如何工作?

    JavaScript尾调用优化(TCO)是复用栈帧避免溢出的技术,要求严格模式、尾位置调用、无arguments/caller/callee引用、静态可确定目标,但主流引擎均未实际支持,应优先使用循环替代。 JavaScript 的尾调用优化(Tail Call Optimization,TCO)是…

    2025年12月21日
    000
  • 如何操作文件_javascript中file api怎么用?

    JavaScript File API 用于浏览器端读取处理用户选择的本地文件,核心是通过 input.files 获取 FileList 对象和 FileReader 异步读取文件内容,支持文本、DataURL、ArrayBuffer 等多种解析方式。 JavaScript 中的 File API…

    2025年12月21日
    000
  • JavaScript与jQuery实现基于JSON数据的动态关联下拉菜单教程

    本教程详细指导如何使用JavaScript和jQuery,结合JSON数据实现动态关联下拉菜单。文章涵盖了JSON数据解析的正确方法、jQuery `$(document).ready()`的最佳实践、动态添加下拉选项、以及如何监听父级下拉菜单的`change`事件来根据选择内容更新子级下拉菜单。通…

    2025年12月21日
    000
  • Next.js应用中实现版本控制的LocalStorage自动清理策略

    ;}export default MyApp; 注意事项与最佳实践 版本号管理: 自动化: 理想情况下,CURRENT_APP_VERSION应该与您的CI/CD流程集成,例如从package.json的版本号中读取,或者在构建时自动注入。这样可以避免手动更新版本号的遗漏。语义化版本: 遵循语义化版…

    2025年12月21日
    000
  • React中子组件向父组件传递状态:以倒计时组件为例实现父组件条件渲染

    本教程详细讲解了如何在react中实现子组件向父组件传递状态。通过“状态提升”模式,父组件将状态更新函数作为props传递给子组件,子组件在特定条件(如倒计时结束)下调用此函数,从而更新父组件的状态。这使得父组件能够根据子组件的内部状态(如计时是否结束)灵活地控制自身的渲染逻辑。 在React应用开…

    2025年12月21日
    000
  • React应用生产环境.env变量读取异常排查与解决方案

    本文旨在解决react应用在生产构建后,`.env`文件中定义的api或配置变量无法正确读取(显示为`null`)的问题。我们将探讨react环境变量的工作原理,分析常见的导致生产环境变量失效的原因,并提供一个具体的解决方案——通过在访问`process.env`变量时使用括号包裹来确保其正确解析,…

    2025年12月21日
    000
  • JavaScript中将对象内嵌套数组数据转换为扁平化格式的教程

    本教程详细介绍了如何使用javascript将包含嵌套数组的对象数据结构,转换为一个扁平化的新数组。通过运用`map()`方法和模板字面量,我们将展示如何高效地从源数据中提取并格式化信息,生成如`”名称 (数量)”`这样简洁易读的字符串数组,从而实现数据的灵活重组和展示。 理…

    2025年12月21日
    000
  • React中利用Map函数动态设置背景图片指南

    本教程旨在解决react应用中使用map函数动态加载背景图片时常见的显示问题。核心在于理解map函数应被用于为数组中的每个数据项生成一个独立的react元素,而非将所有图片路径聚合成一个字符串应用于单个元素。文章将详细阐述错误用法、提供正确的实现代码,并强调在列表渲染中key属性的重要性,确保背景图…

    2025年12月21日
    000
  • 如何用JavaScript实现拖放功能_需要处理哪些鼠标事件?

    JavaScript手动拖放核心是mousedown记录偏移并绑定document的mousemove和mouseup,mousemove时用偏移量更新元素位置,mouseup时解绑事件并重置状态。 用 JavaScript 实现拖放功能,核心是监听并协调一系列鼠标事件,让元素能被“抓起”、随鼠标移…

    2025年12月21日
    000
  • GA4 gtag purchase事件中动态构建items参数的专业指南

    本教程详细阐述了如何在google analytics 4的`gtag` `purchase`事件中,正确地动态构建复杂的`items`参数。针对开发者常犯的字符串拼接错误,文章强调了使用原生javascript对象和数组来组装数据的正确方法,并提供了详细的代码示例。通过遵循本指南,您可以确保电商数…

    2025年12月21日
    000
  • 深入理解HTMLElement.style与CSS自定义属性:短属性展开的陷阱

    本文深入探讨了在javascript中通过`htmlelement.style`访问包含css自定义属性(`var()`)的短属性时,可能遇到的值被错误展开或返回空字符串的问题。文章阐明了`htmlelement.style`仅反映元素的内联样式字面值,无法在解析短属性时预知`var()`的最终解析…

    2025年12月21日
    000
  • Leaflet地图动态标记的正确移除方法:避免常见陷阱

    本教程详细解析了在leaflet应用中动态添加的地图标记无法正确移除的常见问题。核心原因在于尝试移除单个标记变量而非管理所有标记的数组。文章将提供一个有效解决方案,通过遍历存储所有标记的数组并对每个标记实例调用`remove()`方法,确保标记能够从地图上彻底清除,并强调了正确的标记管理实践。 引言…

    2025年12月21日
    000
  • 如何在Terser压缩中避免移除由HTML调用的JavaScript函数

    当使用Terser在模块模式下压缩JavaScript代码时,仅在HTML中调用的函数可能会被意外移除,即使设置了`dead_code: false`。本文将深入解析Terser的优化机制,并提供一个确保此类函数在压缩后依然可用的有效解决方案:通过显式将其绑定到全局`window`对象,从而使其被T…

    2025年12月21日
    000
  • javascript代码性能如何优化_有哪些常见的性能陷阱?

    JavaScript性能优化核心是减少计算、避免主线程阻塞、合理管理内存与资源;需避免频繁DOM操作、隐式类型转换、闭包内存泄漏及加载时机不当等问题。 JavaScript性能优化核心是减少不必要的计算、避免阻塞主线程、合理管理内存和资源。常见陷阱往往藏在看似无害的写法里,而不是大段复杂逻辑中。 避…

    2025年12月21日
    000
  • JavaScript中的this关键字指向什么_不同场景下它的行为有何不同?

    this 指向函数调用时的执行上下文对象,取决于调用方式而非定义位置;全局非严格模式下指向 window 或 global,严格模式及 ESM 中为 undefined;普通调用时非严格模式指向全局对象、严格模式为 undefined;方法调用时指向点号左侧对象;箭头函数无 this,继承外层词法作…

    2025年12月21日
    000
  • 监听HTML数值输入框步进器箭头的点击事件

    本文详细介绍了如何通过JavaScript的change事件来检测HTML type=”number”输入框中步进器箭头(stepper arrows)的点击行为。文章将阐述change事件的工作原理,提供实际代码示例,并探讨如何在步进器点击后实现自定义的数值增减逻辑,尤其适…

    2025年12月21日
    000
  • JavaScript微任务是什么_它何时执行?

    微任务是异步操作中优先级最高的任务,包括Promise回调、MutationObserver、queueMicrotask等,总在当前宏任务结束后、下一个宏任务开始前清空执行,易阻塞渲染且需防无限递归。 JavaScript微任务是异步操作中优先级最高的一类任务,它们会在当前同步代码执行完后、下一次…

    2025年12月21日
    000

发表回复

登录后才能评论
关注微信