JavaScript:从对象数组中提取并保留唯一键值对

javascript:从对象数组中提取并保留唯一键值对

本教程旨在详细阐述如何在JavaScript中高效处理包含重复键值对的对象数组。通过采用`reduce`方法结合一个`seen`映射表来追踪已处理的键值对,我们可以有效地过滤掉后续对象中出现的重复项。文章将提供清晰的算法思路、具体的代码实现及使用示例,帮助开发者构建一个新数组,其中每个对象仅包含首次出现的独特键值对,从而实现数据的去重和结构优化。

引言:理解问题与目标

在JavaScript开发中,我们经常会遇到需要处理包含复杂数据的数组,其中每个元素都是一个对象。有时,这些对象之间可能存在键值对的重复。例如,在一个对象数组中,如果第一个对象包含”Param1″: “20”,而后续某个对象也包含”Param1″: “20”,我们的目标是移除后续对象中的这个重复键值对,只保留第一次出现的。这要求我们遍历数组,并对每个对象的键值对进行去重判断,确保最终结果中的每个键值对都是全局唯一的(即在之前的对象中未出现过)。

考虑以下示例输入:

const arr1 = [  { "Param1": "20", "Param2": "8", "Param3": "11", "Param4": "4", "Param5": "18", "Param6": "20", "Param7": "8" },  { "Param6": "21", "Param7": "8", "Param8": "11", "Param9": "4", "Param10": "18" },  { "Param1": "20", "Param2": "8", "Param3": "10" }];

我们期望的输出是:

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

[  { "Param1": "20", "Param2": "8", "Param3": "11", "Param4": "4", "Param5": "18", "Param6": "20", "Param7": "8" },  { "Param6": "21", "Param8": "11", "Param9": "4", "Param10": "18" }, // "Param7": "8" 被移除,因为它在第一个对象中已出现  { "Param3": "10" } // "Param1": "20" 和 "Param2": "8" 被移除,因为它们在第一个对象中已出现];

注意,去重是基于key和value的组合,并且是顺序敏感的——即如果一个键值对在数组中较早的对象中出现过,它在后续对象中就会被移除。

核心算法思路

为了实现上述目标,我们需要一个机制来追踪所有已经遇到的键值对。我们可以使用一个嵌套的映射结构(或JavaScript对象)来充当这个“已见”集合。

具体算法步骤如下:

初始化追踪器: 创建一个名为 seen 的空对象(或 Map),其结构为 { [key: string]: { [value: string]: boolean } }。这个结构将用于记录某个 key 对应某个 value 是否已经出现过。初始化结果数组: 创建一个空的 result 数组,用于存放处理后的对象。遍历输入数组: 逐个处理输入数组中的每个对象。处理当前对象: 对于当前对象,遍历其所有的键值对。对于每个 [key, value] 对:首先,确保 seen 对象中存在 key 对应的内层对象。如果不存在,则初始化为 {}。检查 seen[key][value] 是否为 true。如果为 true,表示这个 key: value 对之前已经出现过,则忽略它(不将其添加到当前正在构建的新对象中)。如果为 false 或 undefined,表示这个 key: value 对是首次出现。将其添加到当前正在构建的新对象中,并在 seen[key][value] 中标记为 true。构建新对象并添加: 将所有通过筛选的键值对重新组合成一个新的对象,并将其添加到 result 数组中。返回结果: 遍历完成后,返回 result 数组。

示例代码实现

在JavaScript中,Array.prototype.reduce() 方法非常适合这种需要遍历数组并累积结果的场景。我们将使用 reduce 来维护 seen 追踪器和 result 数组。

/** * 从对象数组中移除重复的键值对。 * 如果一个键值对在数组中较早的对象中出现过,则在后续对象中将其移除。 * * @param arr 包含待处理对象的数组。每个对象都是一个键值对集合。 * @returns 包含唯一键值对的新对象数组。 */const removeDuplicates = (arr: Record[]) => {    // 使用 reduce 方法迭代数组并累积结果    return arr.reduce<{        seen: Record<string, Record>; // 追踪已见键值对的映射        result: Record[]; // 存储处理后对象的数组    }>(        (accumulator, currentItem) => {            // 为当前对象创建一个新的对象,只包含唯一的键值对            const uniqueItem = Object.fromEntries(                // 遍历当前对象的键值对,并进行过滤                Object.entries(currentItem).filter(([key, value]) => {                    // 确保 seen[key] 存在,如果不存在则初始化为空对象                    accumulator.seen[key] = accumulator.seen[key] ?? {};                    // 检查当前键值对是否已在 seen 映射中标记为 true                    if (accumulator.seen[key][value]) {                        // 如果已存在,则过滤掉此键值对 (返回 false)                        return false;                    }                    // 如果是首次出现,则标记为已见 (设置为 true)                    accumulator.seen[key][value] = true;                    // 保留此键值对 (返回 true)                    return true;                }),            );            // 将处理后的唯一对象添加到结果数组中            accumulator.result.push(uniqueItem);            return accumulator;        },        // reduce 的初始值:一个包含空 seen 映射和空 result 数组的对象        { seen: {}, result: [] },    ).result; // 最后返回累加器中的 result 数组};// 示例使用const arr1 = [  { "Param1": "20", "Param2": "8", "Param3": "11", "Param4": "4", "Param5": "18", "Param6": "20", "Param7": "8" },  { "Param6": "21", "Param7": "8", "Param8": "11", "Param9": "4", "Param10": "18" },  { "Param1": "20", "Param2": "8", "Param3": "10" }];const uniqueArr = removeDuplicates(arr1);console.log(uniqueArr);/*预期的输出:[  { "Param1": "20", "Param2": "8", "Param3": "11", "Param4": "4", "Param5": "18", "Param6": "20", "Param7": "8" },  { "Param6": "21", "Param8": "11", "Param9": "4", "Param10": "18" },  { "Param3": "10" }]*/

代码解析:

removeDuplicates 函数: 接收一个对象数组作为参数。arr.reduce(…): 这是核心的迭代和累积过程。accumulator (累加器): 初始值为 { seen: {}, result: [] }。seen 对象用于存储已见的键值对,result 数组用于存储最终处理后的对象。currentItem: 当前正在处理的数组中的对象。Object.entries(currentItem): 将当前对象转换为 [key, value] 对的数组。.filter(([key, value]) => { … }): 对 [key, value] 对进行过滤。accumulator.seen[key] = accumulator.seen[key] ?? {};:这是ES2020的空值合并运算符。它确保 accumulator.seen[key] 存在。如果 accumulator.seen[key] 是 null 或 undefined,则将其初始化为一个空对象 {}。if (accumulator.seen[key][value]) return false;:检查当前 key: value 对是否已经在 seen 映射中被标记为 true。如果是,则表示它已出现过,filter 返回 false 将其移除。accumulator.seen[key][value] = true;:如果当前 key: value 对是首次出现,则在 seen 映射中将其标记为 true,表示已见过。return true;:首次出现的键值对通过 filter 筛选。Object.fromEntries(…): 将过滤后的 [key, value] 对数组重新组合成一个新的对象 uniqueItem。accumulator.result.push(uniqueItem);: 将新生成的 uniqueItem 添加到结果数组中。return accumulator;: 返回更新后的累加器,供下一次迭代使用。.result: reduce 方法执行完毕后,我们从最终的累加器中提取 result 数组,即为我们所需的结果。

注意事项与扩展

顺序敏感性: 此方案严格遵循输入数组的顺序。如果输入数组的顺序改变,输出结果也可能不同,因为“首次出现”的定义是基于遍历顺序的。值类型: 示例中键值对的值是字符串。对于其他原始数据类型(如数字、布尔值),此方法同样适用,因为它们在JavaScript中可以直接作为对象键或属性值进行比较。但对于非原始值(如对象、数组),value 的比较将是引用比较,而不是深层内容比较。如果需要深层比较,则需要对 value 进行序列化(例如 JSON.stringify(value))或实现自定义的深层比较逻辑。性能考量:时间复杂度:对于一个包含 N 个对象的数组,每个对象平均有 M 个键值对。reduce 迭代 N 次,每次迭代中 Object.entries 和 filter 都会遍历 M 个键值对。在 filter 内部,对象属性访问(如 seen[key][value])通常是 O(1) 操作。因此,整体时间复杂度大致为 O(N * M)。空间复杂度:seen 对象可能存储所有唯一的键值对,最坏情况下,如果所有键值对都唯一,则空间复杂度为 O(N * M)。TypeScript 类型: 示例代码使用了 TypeScript 类型注解 (Record[]),这有助于提高代码的可读性和健壮性。在纯 JavaScript 环境中,可以直接移除类型注解。

总结

通过本教程,我们学习了一种在JavaScript中从对象数组中提取并保留唯一键值对的有效方法。利用 Array.prototype.reduce() 结合一个嵌套的 seen 对象作为追踪器,我们能够灵活且高效地处理复杂的数据去重需求。这种方法不仅清晰地解决了特定问题,也展示了 reduce 在数据转换和聚合方面的强大能力。在实际开发中,理解其顺序敏感性和对不同值类型的处理方式,将帮助我们更好地应用此模式。

以上就是JavaScript:从对象数组中提取并保留唯一键值对的详细内容,更多请关注创想鸟其它相关文章!

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

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月21日 12:40:56
下一篇 2025年12月21日 12:41:12

相关推荐

  • 前端代码辅助工具:如何选择最可靠的AI工具?

    前端代码辅助工具:可靠性探讨 对于前端工程师来说,在HTML、CSS和JavaScript开发中借助AI工具是司空见惯的事情。然而,并非所有工具都能提供同等的可靠性。 个性化需求 关于哪个AI工具最可靠,这个问题没有一刀切的答案。每个人的使用习惯和项目需求各不相同。以下是一些影响选择的重要因素: 立…

    2025年12月24日
    300
  • TypeScript 中如何约束对象为 CSS 属性?

    typescript 中如何约束对象为 css 属性 想要约束一个对象为 css 属性,以便在调用函数时得到自动补全提示,可以采用以下方法: 使用 react 的 cssproperties 类型 对于 react 项目,可以使用 react 提供的 cssproperties 类型: 立即学习“前…

    2025年12月24日
    300
  • 如何在 TypeScript 中约束对象为 CSS 属性?

    如何在 typescript 中约束对象为 css 属性? 在 typescript 中,为特定目的而约束对象类型是很重要的。在本文中,我们将探究如何将对象约束为包含 css 属性。 考虑以下函数: function setattrstoelement(el: htmlelement, attr: …

    2025年12月24日
    000
  • 如何使用 TypeScript 约束对象以匹配 CSS 属性?

    如何约束 typescript 对象以匹配 css 属性? setattrstoelement 函数接收两个参数,其中第二个参数应为 css 属性。对于 react 项目,可以使用 cssproperties 类型: import { cssproperties } from “react”;fun…

    2025年12月24日
    000
  • 为什么使用 :global 修改 Antd 样式无效?

    :global 修改 antd 样式为何无效 本文旨在帮助您解决在组件内使用:global修改 antd 全局样式未生效的问题。 问题描述 您在组件内使用:global修改 antd 按钮样式,但没有生效。完整代码可参考 https://codesandbox.io/s/fk7jnl 。 解决方案 …

    2025年12月24日
    000
  • 为什么在 React 组件中无法获得 Tailwind CSS 语法提示?

    为什么在 React 组件中无法获得 Tailwind CSS 语法提示? 你在 VSCode 中编写 HTML 文件时,可以正常获取 Tailwind CSS 语法提示。但当你尝试在 React 组件中编写 Tailwind CSS 时,这些提示却消失不见了。这是什么原因造成的? 解决方案 要解决…

    2025年12月24日
    000
  • 如何在 VSCode 中为 React 组件启用 Tailwind CSS 提示?

    在 vscode 中为 react 组件启用 tailwind css 提示 如果你在使用 vscode 编写 react 组件时,发现 tailwind css 提示无法正常显示,这里有一个解决方法: 安装 tailwind css intellisense 插件 这是实现代码提示的关键,确保你已…

    2025年12月24日
    200
  • 如何直接访问 Sass 地图变量的值?

    直接访问 sass 地图变量的值 在 sass 中,我们可以使用地图变量来存储一组键值对。而有时候,我们可能需要直接访问其中的某个值。 可以通过 map-get 函数直接从地图中获取特定的值。语法如下: map-get($map, $key) 其中: $map 是我们要获取值的 sass 地图变量。…

    2025年12月24日
    000
  • CSS 砌体 Catness

    css 就像技术中的其他东西一样 – 它总是在变化和发展。该领域正在进行的开发是 css 网格布局模块级别 3,也称为 css masonry 布局。 theo 制作了一段视频,介绍了它的开发方式以及苹果和谷歌就如何实施它进行的辩论。 所有这些让我很高兴尝试 css 砌体! webkit…

    好文分享 2025年12月24日
    000
  • 我如何编写 CSS 选择器

    CSS 方法有很多,但我都讨厌它们。有些多(顺风等),有些少(BEM、OOCSS 等)。但归根结底,它们都有缺陷。 当然,人们使用这些方法有充分的理由,并且解决的许多问题我也遇到过。因此,在这篇文章中,我想写下我自己的关于如何保持 CSS 井井有条的指南。 这并不是一个任何人都可以开始使用的完整描述…

    2025年12月24日
    000
  • 深入理解CSS框架与JS之间的关系

    深入理解CSS框架与JS之间的关系 在现代web开发中,CSS框架和JavaScript (JS) 是两个常用的工具。CSS框架通过提供一系列样式和布局选项,可以帮助我们快速构建美观的网页。而JS则提供了一套功能强大的脚本语言,可以为网页添加交互和动态效果。本文将深入探讨CSS框架和JS之间的关系,…

    2025年12月24日
    000
  • 项目实践:如何结合CSS和JavaScript打造优秀网页的经验总结

    项目实践:如何结合CSS和JavaScript打造优秀网页的经验总结 随着互联网的快速发展,网页设计已经成为了各行各业都离不开的一项技能。优秀的网页设计可以给用户留下深刻的印象,提升用户体验,增加用户的黏性和转化率。而要做出优秀的网页设计,除了对美学的理解和创意的运用外,还需要掌握一些基本的技能,如…

    2025年12月24日
    200
  • 学完HTML和CSS之后我应该做什么?

    网页开发是一段漫长的旅程,但是掌握了HTML和CSS技能意味着你已经赢得了一半的战斗。这两种语言对于学习网页开发技能来说非常重要和基础。现在不可或缺的是下一个问题,学完HTML和CSS之后我该做什么呢? 对这些问题的答案可以分为2-3个部分,你可以继续练习你的HTML和CSS编码,然后了解在学习完H…

    2025年12月24日
    000
  • 聊聊怎么利用CSS实现波浪进度条效果

    本篇文章给大家分享css 高阶技巧,介绍一下如何使用css实现波浪进度条效果,希望对大家有所帮助! 本文是 CSS Houdini 之 CSS Painting API 系列第三篇。 现代 CSS 之高阶图片渐隐消失术现代 CSS 高阶技巧,像 Canvas 一样自由绘图构建样式! 在上两篇中,我们…

    2025年12月24日 好文分享
    200
  • 巧用距离、角度及光影制作炫酷的 3D 文字特效

    如何利用 css 实现3d立体的数字?下面本篇文章就带大家巧用视觉障眼法,构建不一样的 3d 文字特效,希望对大家有所帮助! 最近群里有这样一个有意思的问题,大家在讨论,使用 CSS 3D 能否实现如下所示的效果: 这里的核心难点在于,如何利用 CSS 实现一个立体的数字?CSS 能做到吗? 不是特…

    2025年12月24日 好文分享
    000
  • CSS高阶技巧:实现图片渐隐消的多种方法

    将专注于实现复杂布局,兼容设备差异,制作酷炫动画,制作复杂交互,提升可访问性及构建奇思妙想效果等方面的内容。 在兼顾基础概述的同时,注重对技巧的挖掘,结合实际进行运用,欢迎大家关注。 正文从这里开始。 在过往,我们想要实现一个图片的渐隐消失。最常见的莫过于整体透明度的变化,像是这样: 立即学习“前端…

    2025年12月24日 好文分享
    000
  • css实现登录按钮炫酷效果(附代码实例)

    今天在网上看到一个炫酷的登录按钮效果;初看时感觉好牛掰;但是一点一点的抛开以后发现,并没有那么难;我会将全部代码贴出来;如果有不对的地方,大家指点一哈。 分析 我们抛开before不谈的话;其实原理和就是通过背景大小以及配合位置达到颜色渐变的效果。 text-transform: uppercase…

    2025年12月24日
    000
  • CSS flex布局属性:align-items和align-content的区别

    在用flex布局时,发现有两个属性功能好像有点类似:align-items和align-content,乍看之下,它们都是用于定义flex容器中元素在交叉轴(主轴为flex-deriction定义的方向,默认为row,那么交叉轴跟主轴垂直即为column,反之它们互调,flex基本的概念如下图所示)…

    2025年12月24日 好文分享
    000
  • 手把手教你用 transition 实现短视频 APP的点赞动画

    怎么使用纯 css 实现有趣的点赞动画?下面本篇文章就带大家了解一下巧妙借助 transition实现点赞动画的方法,希望对大家有所帮助! 在各种短视频界面上,我们经常会看到类似这样的点赞动画: 非常的有意思,有意思的交互会让用户更愿意进行互动。 那么,这么有趣的点赞动画,有没有可能使用纯 CSS …

    2025年12月24日 好文分享
    000
  • 巧用CSS实现各种奇形怪状按钮(附代码)

    本篇文章带大家看看怎么使用 CSS 轻松实现高频出现的各类奇形怪状按钮,希望对大家有所帮助! 怎么样使用 CSS 实现一个内切角按钮呢、怎么样实现一个带箭头的按钮呢? 本文基于一些高频出现在设计稿中的,使用 css 实现稍微有点难度和技巧性的按钮,讲解使用 css 如何尽可能的实现它们。【推荐学习:…

    2025年12月24日 好文分享
    000

发表回复

登录后才能评论
关注微信