JavaScript 中的Set从基础到高级用法

 更新时间:2025年11月05日 10:16:33   作者:风茫  
本文全面介绍了ES6中Set数据结构的特点与用法,文章详细讲解了Set的创建、增删查API、遍历方法,并与Array进行性能对比,本文结合实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧

前言

在前端开发中,我们经常需要处理去重、集合运算、唯一性校验等需求。过去我们通常使用 Array 配合 filter + indexOfreduce 来实现,但代码冗长且性能不佳。

为了解决这类问题,ES6 引入了 Set 数据结构。

本文将带你全面掌握 Set定义、特性、API、使用场景、性能对比与最佳实践,助你在项目中高效处理“唯一性”问题。

一、什么是 Set?

定义

Set 是一个 唯一值的集合,它允许存储任何类型的值,且每个值在集合中只出现一次。

const set = new Set();

核心特性

特性说明
自动去重重复添加的值会被忽略
大小可查set.size 直接获取元素数量
有序存储按插入顺序遍历
无原型污染不继承 Object.prototype,更“干净”
性能更优查找、插入、删除操作平均时间复杂度为 O(1)

二、Set 的基本用法

1. 创建 Set

// 空 Set
const set = new Set();
// 从数组初始化(自动去重)
const set = new Set([1, 2, 3, 2, 1]); // Set { 1, 2, 3 }
// 从字符串创建(字符去重)
const charSet = new Set('hello'); // Set { 'h', 'e', 'l', 'o' }

2. 增删查 API

方法说明返回值
add(value)添加值Set 本身(可链式调用)
has(value)判断值是否存在boolean
delete(value)删除值boolean(是否删除成功)
clear()清空所有void
const userSet = new Set();
userSet.add('Alice');
userSet.add('Bob');
userSet.add('Alice'); // 重复,忽略
console.log(userSet.has('Alice')); // true
console.log(userSet.delete('Bob')); // true
console.log(userSet.size); // 1
userSet.clear();
console.log(userSet.size); // 0

3. 链式调用

new Set()
  .add(1)
  .add(2)
  .add(3);

三、Set 的遍历方式

Set 提供了三种遍历方法,返回 迭代器(Iterator)

方法返回值说明
keys()值的迭代器values() 相同(兼容 Map)
values()值的迭代器默认遍历方式
entries()键值对的迭代器每个元素是 [value, value]

1.for...of遍历

const set = new Set(['a', 'b', 'c']);
// 遍历值
for (const value of set) {
  console.log(value);
}
// a
// b
// c
// 或使用 values()
for (const value of set.values()) {
  console.log(value);
}
// 遍历键值对(每个键值相同)
for (const [key, value] of set.entries()) {
  console.log(key, value); // a a, b b, c c
}

2.forEach遍历

set.forEach((value, key) => {
  console.log(value); // 注意:key === value
});

注意:forEach 的参数中 keyvalue 相同,这是为了与 Map 保持一致。

四、Set 与 Array 的对比

特性SetArray
唯一性自动去重允许重复
大小set.sizearr.length
顺序插入顺序插入顺序
查找性能O(1)O(n)
序列化需手动处理JSON.stringify() 支持
语法set.has(value)arr.includes(value)

何时使用 Set?

场景推荐
去重Set(最优雅)
频繁查找是否存在Set(性能优)
需要保持唯一性Set
需要索引访问(arr[0]Array
需要 mapfilter 等方法Array(需转换)
数据量小、操作简单可任选

五、Set 的高级用法

1. 数组去重(最简洁写法)

const arr = [1, 2, 2, 3, 3, 4];
const uniqueArr = [...new Set(arr)];
// [1, 2, 3, 4]
// 或
const uniqueArr = Array.from(new Set(arr));

2. 集合运算(交集、并集、差集)

const a = new Set([1, 2, 3]);
const b = new Set([2, 3, 4]);
// 并集
const union = new Set([...a, ...b]); // {1, 2, 3, 4}
// 交集
const intersection = new Set([...a].filter(x => b.has(x))); // {2, 3}
// 差集(a - b)
const difference = new Set([...a].filter(x => !b.has(x))); // {1}

3. 数组元素唯一性校验

const hasDuplicates = (arr) => {
  return new Set(arr).size !== arr.length;
};
hasDuplicates([1, 2, 2]); // true
hasDuplicates([1, 2, 3]); // false

4. 转换为 Array / Object

const set = new Set(['a', 'b', 'c']);
// Set → Array
const arr = [...set]; // ['a', 'b', 'c']
// 或 Array.from(set)
// Array → Set
const newSet = new Set(arr);

六、WeakSet:更轻量的 Set

什么是 WeakSet?

  • 值必须是 对象
  • 值是 弱引用,不会阻止垃圾回收
  • 不能遍历,没有 sizeclear()
  • 用于对象标记、私有数据、缓存
const ws = new WeakSet();
const obj = { name: 'Alice' };
ws.add(obj);
console.log(ws.has(obj)); // true
// 当 obj 被销毁,WeakSet 中的引用也会被自动清除

与 Set 对比

特性SetWeakSet
值类型任意对象
弱引用不支持支持
可遍历支持不支持
size / clear()支持不支持
用途通用唯一集合对象标记、缓存、私有数据

七、常见问题

1.Set和Array去重的区别?

答:

  • Set 去重基于 SameValueZero 算法,NaN === NaN
  • ArrayindexOf 使用 ===NaN !== NaN,无法去重 NaN
  • Set 性能更好,代码更简洁。

2.Set如何判断重复?

答:使用 SameValueZero 比较算法:

  • === 大部分情况
  • 特殊:NaN 被认为相等
const set = new Set([NaN, NaN]);
console.log(set.size); // 1

3.WeakSet有什么用?

答:用于对象的临时标记、私有数据存储、避免内存泄漏。例如:

const disabledElements = new WeakSet();
disabledElements.add(button);
// 当 button 被移除,标记自动消失

4. 如何实现对象内容去重?

答:需手动实现,如将对象序列化为字符串:

const objSet = new Set();
const key = JSON.stringify({ id: 1 });
objSet.add(key);

注意顺序、类型、undefined 处理。

总结

要点说明
核心优势自动去重、查找快(O(1))、有序、可查大小
适用场景数组去重、集合运算、唯一性校验、缓存键
遍历方式for...offorEachvalues/entries
转换方法[...set]Array.from(set)
内存优化WeakSet 用于对象的弱引用场景

到此这篇关于JavaScript 深入理解Set:从基础到高级用法的文章就介绍到这了,更多相关js set用法内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 检测网页是否被外部iframe嵌入的完整解决方案

    检测网页是否被外部iframe嵌入的完整解决方案

    在Web开发中,我们有时会遇到页面内容被未经授权的网站通过 <iframe>嵌入的情况,这种行为可能导致用户体验下降,甚至引发安全风险,本文将从前端检测原理、代码实现、服务器防护策略等维度,提供一套完整的解决方案,需要的朋友可以参考下
    2025-05-05
  • json字符串对象转换代码实例

    json字符串对象转换代码实例

    这篇文章主要介绍了json字符串对象转换代码实例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2019-09-09
  • JS简单去除数组中重复项的方法

    JS简单去除数组中重复项的方法

    这篇文章主要介绍了JS简单去除数组中重复项的方法,涉及javascript针对数组的遍历、判断与运算相关操作技巧,需要的朋友可以参考下
    2016-09-09
  • JS模块化开发之EsModule和Common.js区别

    JS模块化开发之EsModule和Common.js区别

    ES Module和CommonJS是JavaScript中两种主要的模块规范,分别用于不同的场景,它们在语法、加载机制、运行时机等方面有显著区别,这篇文章主要介绍了JS模块化开发之EsModule和Common.js区别的相关资料,需要的朋友可以参考下
    2025-11-11
  • JS输入用户名自动显示邮箱后缀列表的方法

    JS输入用户名自动显示邮箱后缀列表的方法

    这篇文章主要介绍了JS输入用户名自动显示邮箱后缀列表的方法,可实现自动显示邮箱后缀名列表的功能,非常具有实用价值,需要的朋友可以参考下
    2015-01-01
  • 简单谈谈gulp-changed插件

    简单谈谈gulp-changed插件

    这篇文章主要介绍了关于gulp-changed插件的相关资料,文中介绍的还是相对详细的,需要的朋友可以参考借鉴,下面来一起看看吧。
    2017-02-02
  • 微信小程序对接七牛云存储的方法

    微信小程序对接七牛云存储的方法

    本篇文章主要介绍了小程序对接七牛云存储的方法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-07-07
  • 基于HTML5上使用iScroll实现下拉刷新,上拉加载更多

    基于HTML5上使用iScroll实现下拉刷新,上拉加载更多

    本文主要介绍在HTML5中使用iScroll实现下拉刷新,上拉加载更多数据的方法,主要就是写了两个自定义函数pullDownAction和pullUpAction,分别在下拉和上拉的事件中调用他们。
    2016-05-05
  • javascript中全局对象的parseInt()方法使用介绍

    javascript中全局对象的parseInt()方法使用介绍

    全局对象的parseInt()方法该如何使用,下面为大家详细介绍下,感兴趣的朋友不要错过
    2013-12-12
  • javascript中类的定义方式详解(四种方式)

    javascript中类的定义方式详解(四种方式)

    这篇文章主要介绍了javascript中类的定义方式,结合实例形式较为详细的分析了JavaScript中类的四种定义方式,具有一定参考借鉴价值,需要的朋友可以参考下
    2015-12-12

最新评论