Set与WeakSet集合是什么,Set集合遍历有几种方法
Admin 2022-07-18 群英技术资讯 712 次浏览
这篇文章主要讲解了“Set与WeakSet集合是什么,Set集合遍历有几种方法”,文中的讲解内容简单、清晰、详细,对大家学习或是工作可能会有一定的帮助,希望大家阅读完这篇文章能有所收获。下面就请大家跟着小编的思路一起来学习一下吧。每天都用数组,有没有过一个Moment,担心插入了重复的值?使用Set集合吧!Set拥有特殊的数据结构,保证插入的值永远不会重复。
通过Set.prototype.constructor 构造函数创建Set实例
/*
* 仅实例化:调用构造函数,不传参数
*/
let empty_set = new Set()
/*
* 实例化同时初始化:传入任意iterate对象,将其转换成Set集合
*/
let transfer_set4arr = new Set([1, 2, 3])
// 返回Set(3) {1, 2, 3}
let transfer_set4string = new Set("huilin")
// 返回Set(5) {"h", "u", "i", "l", "n"}
let transfer_set4set = new Set(new Set([5, 6]))
// 返回Set(2) { 5, 6 }
访问 Set.prototype.size属性,返回集合中元素的个数
console.log(empty_set.size) // 0 console.log(transfer_set4arr.size) // 3
调用 Set.prototype.has(value) 方法,判断元素是否存在
// 相比起Array.includes(),Set.has()性能更高,因为专门对成员测试进行了优化
console.log(empty_set.has(1)) // false
console.log(transfer_set4arr.has('h')) // true
想了解更多关于Object.is(),请跳转查看:developer.mozilla.org/zh-CN/docs/…
let n1 = NaN let n2 = NaN console.log(n1 === n2) // 恒等符号判断两者不一致,输出false console.log(Object.is(n1,n2)) // 但Object.is()判断两者是相同的,输出false // Set集合不允许将两个NaN放入集合 let set = new Set() set.add(n1).add(n2) console.log(set.size) // size: 1
而面对复杂数据类型时,主要通过对象的引用进行判断。引用不一致,即便数据结构一致,也认为只不相同,因此能存入Set集合。
let same_value_set = new Set();
// 先存入一个对象
same_value_set.add({num: 0});
// 再存入一个结构一致的新对象
let obj = {num: 0};
same_value_set.add(obj);
// 都能存入成功
console.log(same_value_set.size); // 2
调用Set.prototype.add(value) 方法,向集合追加数据
// add()方法可以追加任意类型的数据,不论是原始值或者是对象引用
let set1 = new Set()
// 由于add()方法始终返回当前实例的引用,所以进行链式调用
set1.add(1).add(2).add(3)
console.log(set1) // Set(3) {1, 2, 3}
// 注意:当add()传入数组时,Set会将数组实例插入集合,而不是数组内的元素
set1.add([4, 5])
console.log(set1) // Set(4) {1, 2, 3, [4, 5]}
调用Set.prototype.delete(value) 方法,移除集合中的元素
// delete()方法返回移除操作是否成功,与.has()方法一样 let success = set1.delete(1) console.log(success) // true
调用Set.prototype.clear() 方法,清空集合
let num_set = new Set([1, 6, 3])
console.log(num_set)
// Set(3) { 1, 6, 3 }
set1.clear()
console.log(num_set)
// Set(0) {}
由于集合没有下标/索引,通常被认为是“无序集合”。但JavaScript会记住元素插入的顺序,所以遍历的时候也按顺序对元素进行迭代。
直接遍历Set集合
let set = new Set([1, 2, 3, 4, 5])
for(let item of set){
console.log(item)
}
// 依次输出:1 2 3 4 5
创建迭代器进行遍历
/*
* 创建迭代器的有三种方式
* Set.prototype.entries()
* Set.prototype.keys()
* Set.prototype.values()
*/
// Set集合只有value而没有key,但为了使得和遍历Map对象相似,Set.entries()创建新的Iterator对象时,每一项的键和值都相等,即[value,value]
for(let [key,value] of set.entries()){
console.log(value)
}
// 依次输出:1 2 3 4 5
// Set.keys()创建新的Iterator对象,返回每一项值
for(let key of set.keys()){
console.log(key)
}
// 依次输出:1 2 3 4 5
// Set.values()和Set.keys()一致,返回每一项的值
for(let value of set.values()){
console.log(value)
}
// 依次输出:1 2 3 4 5
调用Set.prototype.forEach(callbackFn)方法遍历
// forEach(callbackFn) 按照插入顺序调用callbackFn,取出每项值
set.forEach(item => {
console.log(item)
})
// 依次输出:1 2 3 4 5
/* * Set转Array */ let set1 = new Set([1, 2, 3]) // Array.from()方法 let arr1 = Array.from(set1) // 扩展运算符 let arr2 = [...set1] /* * Array转Set */ // 利用Set构造函数 let set = new Set(array)
let set = new Set([1, 2, 4, 4, 2, 5])
console.log(set)
// Set(4) { 1, 2, 4, 5 }
let arr1 = [1, 2, 4]
let arr2 = [1, 5, 6]
// 利用Set集合的特性,集合内的元素都是唯一的
let result = new Set([...set1, ...set2])
console.log(result)
// Set(5) { 1, 2, 4, 5, 6 }
let set1 = new Set([1, 2, 4])
let set2 = new Set([1, 5, 6])
// 返回set1和set2都存在的元素
let result = new Set([...set1].filter(x => set2.has(x)))
console.log(result)
// Set(1) { 1 }
let set1 = new Set([1, 2, 4])
let set2 = new Set([1, 5, 6])
function isMixed(set, subset) {
for (let elem of subset) {
if (set.has(elem)) {
return true;
}
}
return false;
}
console.log(isMixed(set1, set2))
// true
let set1 = new Set([1, 2, 4])
let set2 = new Set([1, 5, 6])
function difference(setA, setB) {
let result = new Set()
for (let elem of setB) {
if(setA.has(elem)){
result.add(elem)
}
}
return result;
}
console.log(difference(set1, set2))
除了Set集合外,ES6还提供了WeakSet和WeakMap。既然集合的名字都叫“Weak(弱)的集合”了,究竟它“弱”在哪里呢?
WeakSet不允许插入原始值,仅支持对象的引用;
let val1 = {id: 1},
val2 = {id: 2}
let ws = new WeakSet()
// 和Set集合一样,WeakSet的值也不重复,同时add()也返回集合实例,所以可以链式操作
ws.add(val1).add(val1).add(val2)
// 不允许插入基础数据类型
ws.add(3)
// 报错:TypeError:Invalid value used in WeakSet
// 但可以先包装成对象后再插入
let val3 = new Number(3)
ws.add(val3)
console.log(ws.has(val3))
// 输出:true
要说弱引用,先看看什么是强引用:
// 声明一个对象
let handsome = {
name: 'huilin',
age: 30
}
// 放入数组
let arr = [1, handsome, 2]
console.log('release before arr length', arr.length) // 3
// 放入Map
let user = {
oid: 10001,
classify: 'Chinese',
staffReference: handsome
}
console.log('release before map length', Object.keys(user).length) // 3
console.log('----')
// 突然把对象置为null
handsome = null
// 强引用的容器中,对象仍然存在没有被回收
console.log('release after arr length', arr.length) // 3
console.log(arr[1]) // { name: 'huilin', age: 30 }
console.log('release after map length', Object.keys(user).length) // 3
console.log(user.staffReference) // { name: 'huilin', age: 30 }
从测试代码看出,除非容器销毁,否则引用的对象一直没有被回收。而所谓弱引用,就是希望容器是根据元素自动伸缩的,一旦对象为null,容器中的引用也跟着回收。
let obj1 = {
name: 'huilin',
age: 30
}
let obj2 = {
name: 'cc',
age: 29
}
let ws1 = new WeakSet()
ws1.add(obj1).add(obj2)
console.log(ws1.has(obj1)) // true
// 不管是从容器操作元素
ws1.delete(obj1)
console.log(ws1.has(obj1)) // false
// 或者是对象自己置为null,都会自动回收
obj2 = null
console.log(ws1.has(obj2)) // false
Reference
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:mmqy2019@163.com进行举报,并提供相关证据,查实之后,将立刻删除涉嫌侵权内容。
猜你喜欢
本文实例为大家分享是如何使用javascript实现拼图游戏。对于拼图游戏,相信很多朋友都有玩过,文本给介绍的JS实现简单的拼图游戏的代码,下编觉得比较有意思,因此分享给大家作参考,下面我们就一起来看看吧。
方法:1、声明一个验证数字的正则表达式,语法为“var a=/^[1-9][0-9]+$/gi;”;2、在if语句中利用test()方法来验证数字,语法为“if(a.test(指定内容)){是数字代码;}else{不是数字代码;}”。
在我们的日常开发中 不免会有很多需要处理数据的方法 本节主要说一说foreach和some的使用不多说把代码编辑器打开
这篇文章主要为大家详细介绍了原生JS实现呼吸轮播图,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
本文将深入 Node 异步和事件循环的底层实现和执行机制。聊聊Node 为什么要使用异步来作为核心编程模型呢?如何实现异步?以及基于事件循环的异步编程模型。
成为群英会员,开启智能安全云计算之旅
立即注册Copyright © QY Network Company Ltd. All Rights Reserved. 2003-2020 群英 版权所有
增值电信经营许可证 : B1.B2-20140078 粤ICP备09006778号 域名注册商资质 粤 D3.1-20240008