### 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除外),不接受其他类型的值作为键名,而且键名所指向的对象,不计入垃圾回收机制。