### Set数据结构 * Set * WeakSet * Map * WeakMap * * * * * #### Set > ES6新增了一个Set数据结构。它类似于数组,但是成员的值都是唯一的,没有重复的值。Set 本身是一个构造函数,用来生成 Set 数据结构。Set构造函数可以接收一个数组或者类似数组的对象来作为参数,用于初始化。 > 向Set加入值的时候,不会发生类型转换,所以5和"5"是两个不同的值。Set内部判断两个值是否不同,使用的算法叫做“Same-value equality”,它类似于精确相等运算符(===),主要的区别是NaN等于自身,而精确相等运算符认为NaN不等于自身。注意⚠️:***在Set内部,两个NaN是相等。另外,两个对象或数组(数组也是对象)总是不相等的。*** > 案例如下: ~~~ let s = new Set([1, 2, 3, 4, 4]); console.log(s) // [1,2,3,4] // 因为Set数据结构不允许内部有重复值 ~~~ > Set结构的实例有以下属性: > Set.prototype.constructor:构造函数,默认就是Set函数。 Set.prototype.size:返回Set实例的成员总数。 ~~~ let s = new Set([1, 2, 3, 4, 4]); console.log(s.size) // 4 ~~~ > Set实例的方法分为两大类:操作方法(用于操作数据)和遍历方法(用于遍历成员)。下面先介绍四个操作方法: > add(value):添加某个值,返回Set结构本身。 delete(value):删除某个值,返回一个布尔值,表示删除是否成功。 has(value):返回一个布尔值,表示该值是否为Set的成员。 clear():清除所有成员,没有返回值。 > 数组的Array.from方法可以将Set结构转为数组。那么就可以通过该方法来为数组去重了;如下: ~~~ let wtf = ["andy","tom","jack","andy"] //注意这个数组中有两个"andy" let hhh = Array.from(new Set(wtf)) console.log(hhh) //["andy","tom","jack"] ~~~ #### WeakSet > WeakSet结构与Set类似,也是不重复的值的集合。但是,它与Set有两个区别。 首先,WeakSet的成员只能是对象,而不能是其他类型的值。 其次,WeakSet中的对象都是弱引用,即垃圾回收机制不考虑WeakSet对该对象的引用,也就是说,如果其他对象都不再引用该对象,那么垃圾回收机制会自动回收该对象所占用的内存,不考虑该对象还存在于WeakSet之中。这个特点意味着,无法引用WeakSet的成员,因此WeakSet是不可遍历的 > WeakSet是一个构造函数,可以使用new命令,创建WeakSet数据结构。 > 作为构造函数,WeakSet可以接受一个数组或类似数组的对象作为参数。该数组的所有成员,都会自动成为WeakSet实例对象的成员。 ~~~ let a = [[1,2], [3,4]]; let s = new WeakSet(a); console.log(s) //WeakSet {[1, 2], [3, 4]} ~~~ > WeakSet结构有以下三个方法: * WeakSet.prototype.add(value):向WeakSet实例添加一个新成员。 * WeakSet.prototype.delete(value):清除WeakSet实例的指定成员。 * WeakSet.prototype.has(value):返回一个布尔值,表示某个值是否在WeakSet实例之中。 #### Map > JavaScript的对象(Object),本质上是键值对的集合(Hash结构),但是传统上只能用字符串当作键。这给它的使用带来了很大的限制。为了解决这个问题,ES6提供了Map数据结构。它类似于对象,也是键值对的集合,但是“键”的范围不限于字符串,各种类型的值(包括对象)都可以当作键。也就是说,Object结构提供了“字符串—值”的对应,Map结构提供了“值—值”的对应,是一种更完善的Hash结构实现。如果你需要“键值对”的数据结构,Map比Object更合适。 作为构造函数,Map也可以接受一个数组作为参数。该数组的成员是一个个表示键值对的数组。 > Map的键实际上是跟内存地址绑定的,只要内存地址不一样,就视为两个键。这就解决了同名属性碰撞(clash)的问题,我们扩展别人的库的时候,如果使用对象作为键名,就不用担心自己的属性与原作者的属性同名。 > 如果Map的键是一个简单类型的值(数字、字符串、布尔值),则只要两个值严格相等,Map将其视为一个键,包括0和-0。另外,虽然NaN不严格相等于自身,但Map将其视为同一个键。 案例如下: ~~~ let ma = new Map([['name', 'andy'],['title', 'haha']]); console.log(ma) //Map {"name" => "andy", "title" => "haha"} ~~~ > Map结构的实例有以下属性和操作方法: * size属性返回Map结构的成员总数。 * set()方法设置key所对应的键值,然后返回整个Map结构。如果key已经有值,则键值会被更新,否则就新生成该键。set方法返回的是Map本身,因此可以采用链式写法。 * get()方法读取key对应的键值,如果找不到key,返回undefined。 * has()方法返回一个布尔值,表示某个键是否在Map数据结构中。 * delete方法删除某个键,返回true。如果删除失败,返回false。 * clear方法清除所有成员,没有返回值。 > Map原生提供三个遍历器生成函数和一个遍历方法: * keys():返回键名的遍历器。 * values():返回键值的遍历器。 * entries():返回所有成员的遍历器。 * forEach():遍历Map的所有成员。 #### WeakMap > WeakMap结构与Map结构基本类似,唯一的区别是它只接受对象作为键名(null除外),不接受其他类型的值作为键名,而且键名所指向的对象,不计入垃圾回收机制。