JS 在以前只有一种集合类型,也就是数组类型。ES6 向 JS 添加了 Set 与 Map集合类型。
7.1 ES6 的 Set
ES6 新增了 Set 类型,这是一种无重复值的有序列表。
let set = new Set();
set.add(5);
set.add("5");
set.add(5); // 会被忽略
console.log(set.size); // 2
console.log(set.has(5)); // true 可以使用 has() 方法来测试某个值是否存在于 Set 中
set.delete(5); // 移除值
console.log(set.has(5)); //false
set.clear(); // set 就被清空了。
console.log(set.has("5")); // false
console.log(set.size); // 0
Set 不会使用强制类型转换来判断值是否重复。
let set = new Set(),
key1 = {},
key2 = {};
set.add(key1);
set.add(key2);
console.log(set.size); // 2
可以使用数组来初始化一个 Set ,并且 Set 构造器会确保不重复地使用这些值
let set = new Set([1, 2, 3, 4, 5, 5, 5, 5]);
console.log(set.size);
7.2 Set 上的 forEach() 方法
forEach() 方法会被传递一个回调函数,该回调接受三个参数:
- Set 中下个位置的值;
- 与第一个参数相同的值;
- 目标 Set 自身。
具有 forEach() 方法的其他对象(即数组与 Map )都会给回调函数传递三个参数,前两个参数都分别是下个位置的值与键(给数组使用的键是数值索引)。
let set = new Set([1, 2]);
set.forEach(function(value, key, ownerSet) {
console.log(key + " " + value);// 1 1 2 2
console.log(ownerSet === set);// true true
});
如果想在回调函数中使用 this ,可以给 forEach() 传入一个 this值作为第二个参数
let set = new Set([1, 2]);
let processor = {
output(value) {
console.log(value); // 1, 2
},
process(dataSet) {
dataSet.forEach(function(value) {
this.output(value);
}, this);
}
}
processor.process(set);
或者
let set = new Set([1, 2]);
let processor = {
output(value) {
console.log(value); // 1, 2
},
process(dataSet) {
dataSet.forEach((value) => this.output(value);
}
}
processor.process(set);
Set无法像数组那样用索引来直接访问某个值。我们可以把Set转换为数组。
7.3 将 Set 转换为数组
数组转化为Set,可以防止值的重复
let set = new Set([1, 2, 3, 3, 3, 4, 5]),
array = [...set];
console.log(array); // [1,2,3,4,5]
7.4 Weak Set
对象存储在 Set 的一个实例中时,实际上相当于把对象存储在变量中。只要对于 Set 实例的引用仍然存在,所存储的对象就无法被垃圾回收机制回收,从而无法释放内存。
let set = new Set(),
key = {};
set.add(key);
key = null;
key = [...set][0]; // {}
// set中还存在{}的引用
let set = new WeakSet(),
key = {};
set.add(key);
key = null;
set.has(key) // false
- 对于 WeakSet 的实例,若调用 add() 方法时传入了非对象的参数,就会抛出错误(
has() 或 delete() 则会在传入了非对象的参数时返回 false ); - Weak Set 不可迭代,因此不能被用在 for-of 循环中;
- Weak Set 无法暴露出任何迭代器(例如 keys() 与 values() 方法),因此没有任何编
程手段可用于判断 Weak Set 的内容; - Weak Set 没有 forEach() 方法;
- Weak Set 没有 size 属性。
7.5 ES6 的 Map
Map是键值对的有序列表,而键和值都可以是任意类型。键的比较使用的是
Object.is()
let map = new Map(), key1 = {},
map.set("title", "Understanding ES6");
map.set(key1, 2016);
console.log(map.size); // 2
console.log(map.get("title")); // "Understanding ES6"
7.6 Map 的方法
- has(key) :判断指定的键是否存在于 Map 中;
- delete(key) :移除 Map 中的键以及对应的值;
- clear() :移除 Map 中所有的键与值。
- get() :获取键值
- map属性
7.7 Map 的初始化
内部数组的首个项会作为键,第二项则为对应值。
let map = new Map([["name", "Nicholas"], ["age", 25]]);
console.log(map.has("name")); // true
console.log(map.get("name")); // "Nicholas"
console.log(map.size); // 2
7.8 Map 上的 forEach 方法
forEach参数
- Map 中下个位置的值;
- 该值所对应的键;
- 目标 Map 自身。
let map = new Map([ ["name", "Nicholas"], ["age", 25]]);
map.forEach(function(value, key, ownerMap) {
console.log(key + " " + value);
console.log(ownerMap === map);
});
//name Nicholas
//true
//age 25
//true
7.9 Weak Map
ES6 的 WeakMap 类型是键值对的无序列表,其中键必须是非空的对象,值则允许是任意类型。
Weak Map 的键才是弱引用,而值不是。在 Weak Map 的值中存储对象会阻止垃圾回收,即使该对象的其他引用已全都被移除。
let map = new WeakMap(),
element = {};
map.set(element, "Original");
let value = map.get(element);
element = null;
Weak Map clear() 方法不存在
里面涉及内存机制。不过,在不涉及键为对象的时候,我们使用一般的map就好,涉及的时候,试试使用Weak Map