在JavaScript中,如何给字符串数组中的元素去重

在 JavaScript 中,给字符串数组(或其他类型数组)去重有多种方法,下面列举几种常用的方式,并说明各自的特点。


1. 利用 Set(最简洁、推荐)

Set 是 ES6 新增的数据结构,它类似于数组,但成员的值都是唯一的。将数组传入 Set 构造函数,再通过扩展运算符或 Array.from() 转回数组即可。

javascript 复制代码
const arr = ['apple', 'banana', 'apple', 'orange', 'banana'];
const unique = [...new Set(arr)];
console.log(unique); // ['apple', 'banana', 'orange']

优点 :代码简洁,效率高(时间复杂度 O(n))。
缺点 :无法保留原数组中的元素顺序吗?------会保留第一次出现的顺序,Set 遍历顺序就是插入顺序,因此顺序是保持的。所以一般没有缺点。但注意 :Set 去重使用"SameValueZero"比较,认为 NaN 与自身相等,而 === 不认为相等;在字符串数组中基本无影响。


2. 利用 filter + indexOf

通过数组自身的 filter 方法,配合 indexOf 检测当前元素第一次出现的位置是否等于当前索引。

javascript 复制代码
const arr = ['apple', 'banana', 'apple', 'orange', 'banana'];
const unique = arr.filter((item, index) => arr.indexOf(item) === index);
console.log(unique); // ['apple', 'banana', 'orange']

优点 :兼容性非常好(ES5 即可),语义清晰。
缺点indexOf 会从头遍历数组,整体时间复杂度 O(n²),在大型数组上性能较差。同时 indexOf 无法正确查找 NaN,但字符串数组通常无此问题。


3. 利用 reduce 累加器

通过 reduce 构建新数组,判断当前元素是否已存在于累加器中。

javascript 复制代码
const arr = ['apple', 'banana', 'apple', 'orange', 'banana'];
const unique = arr.reduce((acc, cur) => {
  if (!acc.includes(cur)) {
    acc.push(cur);
  }
  return acc;
}, []);
console.log(unique); // ['apple', 'banana', 'orange']

优点 :逻辑直观,可灵活扩展。
缺点includes 也是遍历,时间复杂度 O(n²);同时 includes 能处理 NaN 但字符串场景无影响。


4. 利用对象/Map 作为哈希表(性能最优)

使用 Map 或普通对象记录已出现的元素,一次遍历即可,时间复杂度 O(n)。

javascript 复制代码
const arr = ['apple', 'banana', 'apple', 'orange', 'banana'];
const map = new Map();
const unique = arr.filter(item => {
  if (map.has(item)) return false;
  map.set(item, true);
  return true;
});
console.log(unique); // ['apple', 'banana', 'orange']

用对象也可以,但对象键会被转为字符串,对于纯字符串数组没问题,但若数组中包含数字或其他类型,需注意类型转换。

javascript 复制代码
const obj = {};
const unique = arr.filter(item => 
  obj.hasOwnProperty(item) ? false : (obj[item] = true)
);

优点 :线性时间复杂度,适合大数据量。
缺点:代码略复杂。


5. 进阶:保持首次出现顺序 + 去重

上述所有方法中,除了 Setfilter+indexOf 默认保持顺序,其余若稍加调整也能保持顺序。SetMap 本身也是按插入顺序迭代,因此上述基于 SetMap 的方案自然保持顺序。


总结

  • 日常开发 :推荐使用 [...new Set(arr)],代码最简洁,性能也不错。
  • 需要兼容极旧环境(如 IE) :可用 filter+indexOf
  • 大数据量且对性能有极致要求 :用 Map 哈希表方式。

注意,以上方法不仅适用于字符串数组,同样适用于数字、混合类型数组,但去重逻辑是基于严格相等(或 Set 的 SameValueZero)。对于对象数组,则需要根据对象的某个属性去重,以上方法需配合 map 提取属性值再进行去重。

相关推荐
测试_AI_一辰2 小时前
项目实战15:Agent主观题怎么评测?先定底线,再做回归
开发语言·人工智能·功能测试·数据挖掘·ai编程
me8322 小时前
【Java面试】Java核心关键字解析(static_final_访问修饰符)小白易懂
java·开发语言·面试
小飞学编程...2 小时前
【Java相关八股文(一)】
android·java·开发语言
打瞌睡的朱尤2 小时前
Vue day12 Vue3认识,写法区分
前端·javascript·vue.js
阿珊和她的猫2 小时前
Vue Router 的使用指南
前端·javascript·vue.js
前路不黑暗@2 小时前
Java项目:Java脚手架项目的通用组件的封装(七)
java·开发语言·spring boot·后端·学习·spring cloud·maven
打瞌睡的朱尤2 小时前
day8 Vue-x
前端·javascript·vue.js
Web打印2 小时前
Phpask(php集成环境)之04配置网站
开发语言·前端·php
一只大侠的侠2 小时前
React Native for OpenHarmony:Calendar 日历组件实现指南
javascript·react native·react.js