Set、Map、WeakSet 和 WeakMap 的区别

Set(集合)

特点

  • 用于 数组重组 生成Set数据结构
  • 类似于数组
  • 成员 唯一 无序 没有重复的值
  • 本身是 构造函数
  • 加入值的时候不会发生类型转换 即(5和'5'不是同一个值)
  • 判断值是否相等(Same-value-zero equality),类似 === (主要区别:Set认为NaN等于自·身)
let arr = [1,3,2,4,5,9,1,3,6,7,8,3,9];
arr.forEach(val => s.add(val));
s.forEach(val => {console.log(val)}); // 1 3 2 4 5 9 6 7 8
console.log(...s) // 1 3 2 4 5 9 6 7 8
console.log([...s]) // [1, 3, 2, 4, 5, 9, 6, 7, 8]  
    
// 去重数组重复对象
let arr = [1,3,2,4,5,9,1,3,6,7,8,3,9];
console.lg([...new Set(arr)]); // [1, 3, 2, 4, 5, 9, 6, 7, 8]

// NaN等于自身
let a = NaN;
let b = NaN;
let s = new Set();
s.add(a);
s.add(b);
console.log(s); // // Set(1) {NaN}

实例属性

  • constructor:构造函数
  • size:元素数量
let set = new Set([1, 2, 3, 2, 1]);

console.log(set.length);   // undefined
console.log(set.size); // 3

实例方法

  • 操作方法
    • add(val): 新增
    • delete(val): 存在即删除
    • has(val): 判断是否存在
    • clear(): 清空
let set = new Set();
set.add(1).add(2).add(1);

set.has(1);  // true
set.has(3);  // false
set.delete(1);   
set.has(1);  // false
set.clear(); // Set(0) {}
  • 遍历方法
    ○ Array.from方法(...)可以将Set结构转为数组
let items = new Set([1, 2, 3, 2]);

let array = Array.from(items);
console.log(array); // [1, 2, 3]

let arr = [...items];
console.log(arr); // [1, 2, 3]

○ keys(): 返回 包含 所有键 的迭代器 (SetIterator {...})
○ values(): 返回 包含 所有值 的迭代器 (SetIterator {...})
○ entries(): 返回 包含 所有键值对 的迭代器 (SetIterator {...})

let items = new Set([7, 8, 9, 2]);

console.log(items.keys()); // SetIterator {7, 8, 9, 2}
console.log(items.values()); // SetIterator {7, 8, 9, 2}
console.log(items.entries()); // SetIterator {7 => 7, 8 => 8, 9 => 9, 2 => 2}

// 可以用for of编辑迭代器
for (let item of set.keys()) {
  console.log(item);  // 7 8 9 2
}

○ forEach(callbackFn, thisArg): 对集合成员执行callbackFn操作,没有返回值。如果有thisArg参数,回调中的this会是这个参数。
○ 可以默认遍历,默认迭代器生成函数是values()方法
○ 可以使用map、filter方法 ([...set])

let set = new Set([1, 2, 3])
set = new Set([...set].map(item => item * 2))
console.log(set);   // Set(3) {2, 4, 6}

let set2 = [...set].map(item => item * 2)
console.log(set2); // [2, 4, 6]

○ 实现数组交集intersect、并集union、差集difference

let set1 = new Set([1, 2, 3]);
let set2 = new Set([4, 3, 2, 8]);

let intersect = [...set1].filter(val => set2.has(val));
let union = [...new Set([...set1, ...set2])];
let diffrence = [...set1].filter(val => !set2.has(val));

console.log(intersect); //  [2, 3]
console.log(union);  // [1, 2, 3, 4, 8]
console.log(diffrence); // [1]

WeakSet

特点

  • 成员都是对象、都是弱引用 (弱引用是指向地址的变量?)
  • 弱引用 并不会屏蔽 垃圾回收机制的。可以被垃圾回收机制回收
  • 可以用来保存 DOM 节点,不容易造成内存泄漏
  • 不能被遍历的

属性

  • constructor:构造函数。任一个具有Iterable接口的对象,都可以作为参数

方法

一般都是用方法设置成员

  • add(val)
  • has(val)
  • delete(val)
  • clear()

WeakSet 和 Set 区别

  • WeakSet只能存储对象引用,不能存放值
  • WeakSet对象中储存的对象值都是被弱引用的
  • WeakSet不会屏蔽垃圾回收机制
  • 对于一个对象,如果没有其他变量或者属性引用该对象,则这个对象会被垃圾回收掉(注下一点)
  • 垃圾回收机制不考虑对象是否存在于WeakSet
  • WeakSet中有多少元素,取决于垃圾回收机制有没有运行
  • WeakSet没有办法拿到它包含的所有元素

Map(字典)

特点

  • 本质:键值对的集合,类似集合
  • 可以遍历
  • 可以和各种数据结构转换
  • 类似对象
  • 键:任意值(值-值)
  • 键:有序的(FIFO原则)
  • 键的个数:从Map的size属性获取
  • 键:不可以重复,键名冲突,会覆盖原有的值

Object (对象)

  • 本质:键值对的集合
  • 键只能是字符串或者Symbols(字符串-值)
  • 键:无序的
const data = {};
const element = document.getElementById('myDiv');

data[element] = 'metadata';
data['[object HTMLDivElement]']; // "metadata"
// 上述是将一个DOM节点作为data的键,但是由于对象只接受字符串作为建,所以element会被转为字符串[Object HTMLElement]

方法

  • size:属性,取Map长度
  • set(key, value):添加
  • get(key):获取
  • has(key):判断是否存在键key
  • deelte(key):删除key对应的数据
  • clear():将Map中所有数据删除

遍历

  • for...of
  • forEach

Map 与 数组之间的转换

let arr = [[1,2], [3, 4], [5, 6]];
let map = new Map(arr);
let newMap = Array.from(map);
let newMap2 = [...map];
console.log(map);
console.log(newMap);
console.log(newMap2);

WeakMap

特点

  • 键值对的集合
  • 键:弱引用对象(null除外)
  • 值:任意
  • 键名所指向的对象可以被垃圾回收,此时键名是无效的
  • 不能遍历

方法

  • has(key)
  • get(key)
  • set(key)
  • delete(key)

集合 和 字典 区别

  • 共同点
    ○ 都可以存储不重复的值
  • 不同点
    ○ 集合:以[value, value]的形式存储元素
    ○ 字典:以[key, value]的形式存储元素

总结

  • Set
    ○ 成员:唯一、无序、不重复
    ○ [value, value]:键值键名一直,即只有键值
    ○ 方法:add\delete\has\clear
    ○ 可以遍历
  • WeakSet
    ○ 成员:弱引用对象
    ○ 可以被垃圾回收机制回收(可以存储DOM节点,不容易造成内存泄露)
    ○ 方法:add\delete\has\clear
    ○ 不能遍历
  • Map
    ○ 本质:键值对的集合,类似集合
    ○ 方法:size\set\get\has\delete\clear
    ○ 可以遍历
  • WeakMap
    ○ 键:弱引用对象(null除外)
    ○ 值:任意
    ○ 键名所指向的对象可以被垃圾回收(此时键名是无效的)
    ○ 方法:get\set\has\delete\clear
    ○ 不能遍历

你可能感兴趣的:(Set、Map、WeakSet 和 WeakMap 的区别)