0

0

JavaScript/Vuex 中基于指定键值对数组对象进行高效去重

霞舞

霞舞

发布时间:2025-07-22 20:22:00

|

428人浏览过

|

来源于php中文网

原创

javascript/vuex 中基于指定键值对数组对象进行高效去重

本文旨在详细阐述如何在 JavaScript 环境(尤其是在 Vuex 的 computed 属性中)对数组对象进行去重操作,使其仅保留指定键(如 trip_class)值唯一的对象。文章将分析常见错误,并提供基于 reduce 方法和 Map 对象的两种高效解决方案,辅以清晰的代码示例和性能考量,帮助开发者正确实现数组对象的去重逻辑。

引言:处理数组对象去重的问题

在前端开发中,我们经常需要处理包含多个对象的数组,并从中提取出满足特定条件(例如某个属性值唯一)的数据。例如,给定一个航班列表,我们可能需要根据 trip_class(行程类别)去重,只保留每种行程类别中的第一个或任意一个代表性航班。这在展示筛选条件、生成唯一列表等场景中尤为常见。

常见误区与问题分析

在尝试对数组对象进行去重时,一个常见的错误是未能正确比较对象的属性值。考虑以下原始数据结构:

[
    {name: 'john', trip_class: 0, lastname: 'lastname'},
    {name: 'Don', trip_class: 1, lastname: 'lastname'},
    {name: 'Joshua', trip_class: 1, lastname: 'lastname'},
    {name: 'Mary', trip_class: 2, lastname: 'lastname'}
]

我们的目标是得到:

[
    {name: 'john', trip_class: 0, lastname: 'lastname'},
    {name: 'Don', trip_class: 1, lastname: 'lastname'},
    {name: 'Mary', trip_class: 2, lastname: 'lastname'}
]

一个常见的错误实现方式,尤其是在使用 reduce 结合 find 时,可能会写出类似以下的代码:

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

// 错误的实现示例
flightsClasses.reduce((acc, obj)=>{
  // 错误点:obj.trip_class (数字) 与 flightClass (对象) 比较
  var exist = acc.find((flightClass) => obj.trip_class === flightClass );
  if(!exist){
    acc.push(obj);
  }
  return acc;
},[]);

上述代码的问题在于 acc.find((flightClass) => obj.trip_class === flightClass ) 这一行。obj.trip_class 是一个数字(例如 0, 1, 2),而 flightClass 是 acc 数组中的一个完整对象。将一个数字与一个对象进行严格相等(===)比较,结果始终为 false,导致 find 总是找不到匹配项,从而所有对象都被添加到累加器 acc 中,未能实现去重。

正确的比较方式应该是比较 obj.trip_class 与 flightClass.trip_class,即比较两个对象的特定属性值。

解决方案一:使用 reduce 方法进行去重

Array.prototype.reduce() 方法非常适合用于将数组归纳为单个值或新的数组。结合 Array.prototype.some() 或 Array.prototype.find() 方法,我们可以有效地实现去重。

GitHub Copilot
GitHub Copilot

GitHub AI编程工具,实时编程建议

下载
computed: {
    flights() {
        return this.$store.getters.getFlights;
    },
    flightsClasses() {
        const flights = this.flights; // 获取原始航班数据

        // 使用 reduce 方法进行去重
        const uniqueFlights = flights.reduce((accumulator, currentFlight) => {
            // 检查累加器中是否已存在具有相同 trip_class 的航班
            const exists = accumulator.some(
                (existingFlight) => existingFlight.trip_class === currentFlight.trip_class
            );

            // 如果不存在,则将当前航班添加到累加器中
            if (!exists) {
                accumulator.push(currentFlight);
            }
            return accumulator;
        }, []); // 初始累加器为空数组

        return uniqueFlights;
    }
}

代码解析:

  • reduce 方法遍历 flights 数组中的每个 currentFlight。
  • accumulator 是一个在每次迭代中累积结果的数组。
  • accumulator.some(...) 用于检查 accumulator 中是否已经存在一个对象的 trip_class 与 currentFlight.trip_class 相同。some() 方法在找到第一个满足条件的元素后会立即返回 true,效率较高。
  • 如果 exists 为 false(即当前 trip_class 尚未被添加),则将 currentFlight 添加到 accumulator 中。
  • 最终 reduce 方法返回包含唯一 trip_class 对象的数组。

解决方案二:利用 Map 对象进行高效去重

对于大型数据集,或者当需要更灵活地控制去重逻辑(例如保留每个 trip_class 的最后一个出现对象)时,Map 对象提供了一种更高效和简洁的去重方法。Map 对象以键值对的形式存储数据,并且键是唯一的。

computed: {
    flights() {
        return this.$store.getters.getFlights;
    },
    flightsClasses() {
        const flights = this.flights; // 获取原始航班数据
        const uniqueMap = new Map(); // 创建一个新的 Map 对象

        // 遍历航班数据,以 trip_class 作为键,航班对象作为值
        flights.forEach(flight => {
            // 如果 Map 中没有当前 trip_class 的键,或者需要更新(例如保留最后一个)
            // uniqueMap.set(flight.trip_class, flight); // 这会保留最后一个出现的对象

            // 如果需要保留第一个出现的对象(更常见需求),则在键不存在时才添加
            if (!uniqueMap.has(flight.trip_class)) {
                uniqueMap.set(flight.trip_class, flight);
            }
        });

        // 将 Map 的值(即去重后的对象)转换为数组
        return Array.from(uniqueMap.values());
    }
}

代码解析:

  • 我们创建一个空的 Map 对象 uniqueMap。
  • 遍历原始 flights 数组。对于每个 flight 对象,我们使用其 trip_class 作为 Map 的键,flight 对象本身作为值。
  • Map 的键是唯一的。当我们调用 uniqueMap.set(key, value) 时,如果 key 已经存在,则其对应的值会被新值覆盖。
  • 为了实现“保留第一个”的去重逻辑,我们增加了一个 if (!uniqueMap.has(flight.trip_class)) 条件判断,确保只有当 trip_class 键尚未存在时才添加。
  • 最后,Array.from(uniqueMap.values()) 将 Map 中所有值(即去重后的对象)提取出来,并转换为一个新的数组。

Map 方案的优势:

  • 性能优越: Map 的查找、插入、删除操作的平均时间复杂度为 O(1),相比 reduce 方案中每次迭代都可能进行 O(N) 的 some 或 find 操作(总复杂度 O(N^2)),Map 方案的整体时间复杂度为 O(N),对于大型数组性能更佳。
  • 代码简洁: 逻辑更直观,易于理解。

在 Vuex computed 属性中的应用

无论是 reduce 还是 Map 方案,都非常适合在 Vue 组件的 computed 属性中使用。computed 属性是响应式的,当其依赖的数据(例如 this.$store.getters.getFlights)发生变化时,flightsClasses 会自动重新计算,确保界面上展示的数据始终是最新的去重结果。



注意事项与性能考量

  1. 保持数据不可变性: 在 computed 属性中进行数据转换时,始终返回一个新的数组或对象,而不是直接修改原始数据。上述两种方法都创建了新的数组,遵循了不可变性原则。
  2. 选择合适的去重键: 确保你选择的键(如 trip_class)能够唯一标识你想要保留的对象。如果需要根据多个键进行复合去重,可以考虑将多个键的值组合成一个唯一的字符串作为 Map 的键,例如 ${flight.trip_class}-${flight.destination}。
  3. 性能对比:
    • reduce + some (或 find): 简单易懂,适用于小型到中等规模的数组。时间复杂度为 O(N^2),因为 some 在最坏情况下需要遍历累加器中的所有元素。
    • Map: 性能更优,适用于大型数组。时间复杂度为 O(N),因为 Map 的查找和插入操作通常是常数时间。
  4. 保留策略:
    • Map 方案默认(uniqueMap.set(key, value))会保留具有相同键的最后一个出现的对象。
    • Map 方案通过 if (!uniqueMap.has(key)) 条件可以实现保留第一个出现的对象。
    • reduce 方案默认是保留第一个出现的对象,因为一旦找到匹配项就不会再添加。

总结

对数组对象进行去重是 JavaScript 开发中的常见任务。理解常见的错误(如错误的比较条件)是解决问题的第一步。本文提供了两种有效的解决方案:基于 reduce 方法的直观实现,以及基于 Map 对象的高效实现。在实际应用中,根据数组的规模和性能要求,选择最适合的去重策略至关重要。在 Vuex computed 属性中运用这些方法,能够确保数据处理的响应性和高效性。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

775

2023.08.22

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

298

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

212

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1498

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

623

2023.11.24

java读取文件转成字符串的方法
java读取文件转成字符串的方法

Java8引入了新的文件I/O API,使用java.nio.file.Files类读取文件内容更加方便。对于较旧版本的Java,可以使用java.io.FileReader和java.io.BufferedReader来读取文件。在这些方法中,你需要将文件路径替换为你的实际文件路径,并且可能需要处理可能的IOException异常。想了解更多java的相关内容,可以阅读本专题下面的文章。

612

2024.03.22

php中定义字符串的方式
php中定义字符串的方式

php中定义字符串的方式:单引号;双引号;heredoc语法等等。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

587

2024.04.29

go语言字符串相关教程
go语言字符串相关教程

本专题整合了go语言字符串相关教程,阅读专题下面的文章了解更多详细内容。

170

2025.07.29

Python 自然语言处理(NLP)基础与实战
Python 自然语言处理(NLP)基础与实战

本专题系统讲解 Python 在自然语言处理(NLP)领域的基础方法与实战应用,涵盖文本预处理(分词、去停用词)、词性标注、命名实体识别、关键词提取、情感分析,以及常用 NLP 库(NLTK、spaCy)的核心用法。通过真实文本案例,帮助学习者掌握 使用 Python 进行文本分析与语言数据处理的完整流程,适用于内容分析、舆情监测与智能文本应用场景。

10

2026.01.27

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Vue 教程
Vue 教程

共42课时 | 7.3万人学习

Vue3.x 工具篇--十天技能课堂
Vue3.x 工具篇--十天技能课堂

共26课时 | 1.5万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号