14、Set 和 Map 数据结构

文章目录

  • 14、Set 和 Map 数据结构
    • Set
      • 1.1 基本用法
      • ☆☆☆ 值唯一,没有重复的值
      • ☆☆☆ 接受数组、具有 iterable 接口的数据结构
      • ☆☆☆ 数组去重1:[...new Set(array)]
      • ☆☆☆ 字符串去重:[...new Set('ababbc')].join('')
      • 1.2 Set实例的属性和方法
      • -- Array.from() 与 Set
      • ☆☆☆ 数组去重2:Array.from(new Set([1, 2, 3, 4, 5]));
      • 1.3 遍历操作
      • - values(),entries()
      • ☆☆☆ Set 默认遍历器 === values()
      • - values(),entries()

14、Set 和 Map 数据结构

Set

1.1 基本用法

ES6(ES6于2015年6月发布)提供了新的数据结构 Set。它类似于数组,但是成员的值都是唯一的,没有重复的值。

☆☆☆ 值唯一,没有重复的值

Set本身是一个构造函数,用来生成 Set 数据结构。

const s = new Set();

[2, 3, 5, 4, 5, 2, 2].forEach(x => s.add(x));

for (let i of s) {
  console.log(i);
}
// 2 3 5 4

☆☆☆ 接受数组、具有 iterable 接口的数据结构

Set函数可以接受一个数组(或者具有 iterable 接口的其他数据结构)作为参数,用来初始化。

// 例一
const set = new Set([1, 2, 3, 4, 4]);
[...set]
// [1, 2, 3, 4]

// 例二
const items = new Set([1, 2, 3, 4, 5, 5, 5, 5]);
items.size // 5

// 例三
const set = new Set(document.querySelectorAll('div'));
set.size // 56

// 类似于
const set = new Set();
document
 .querySelectorAll('div')
 .forEach(div => set.add(div));
set.size // 56

上面代码中,例一和例二都是Set函数接受数组作为参数,例三是接受类似数组的对象作为参数。

☆☆☆ 数组去重1:[…new Set(array)]

// 去除数组的重复成员
[...new Set(array)]

☆☆☆ 字符串去重:[…new Set(‘ababbc’)].join(‘’)

[...new Set('ababbc')].join('') // Set->Array->String
// "abc"

[...new Set('ababbc')].join(' - ')
// "a - b - c"

向 Set 加入值的时候,不会发生类型转换,所以5"5"是两个不同的值。Set 内部判断两个值是否不同,使用的算法叫做“Same-value-zero equality”,它类似于精确相等运算符(===),主要的区别是向 Set 加入值时认为NaN等于自身,而精确相等运算符认为NaN不等于自身。

let set = new Set();
let a = NaN;
let b = NaN;
set.add(a);
set.add(b);
set // Set {NaN}

————
另外,两个对象总是不相等的。

let set = new Set();

set.add({});
set.size // 1

set.add({});
set.size // 2

1.2 Set实例的属性和方法

Set 结构的实例有以下属性。
Set.prototype.constructor:构造函数,默认就是Set函数。
Set.prototype.size:返回Set实例的成员总数

Set 实例的方法分为两大类:操作方法(用于操作数据)和遍历方法(用于遍历成员)。下面先介绍四个操作方法。

Set.prototype.add(value):添加某个值,返回 Set 结构本身
Set.prototype.delete(value):删除某个值,返回一个布尔值,表示删除是否成功
Set.prototype.has(value):返回一个布尔值,表示该值是否为Set的成员
Set.prototype.clear():清除所有成员,没有返回值

s.add(1).add(2).add(2);   // 添加某个值,返回 Set 结构本身
// 注意2被加入了两次

s.size // 2

s.has(1) // true
s.has(2) // true
s.has(3) // false

s.delete(2) // true
s.has(2) // false

————
下面是一个对比,判断是否包括一个键,Object结构和Set结构写法的不同。

// 对象的写法
const properties = {
  'width': 1,
  'height': 1
};

if (properties[someName]) {  // 对象这样写法,参考对象的拓展1
  // do something
}

// Set的写法
const properties = new Set();

properties.add('width');
properties.add('height');

if (properties.has(someName)) {
  // do something
}

————

– Array.from() 与 Set

☆☆☆ 数组去重2:Array.from(new Set([1, 2, 3, 4, 5]));

Array.from()方法可以将 Set 结构转为数组。–参考 “9、数组的扩展”

const items = new Set([1, 2, 3, 4, 5]);
const array = Array.from(items); // 

这就提供了去除数组重复成员的另一种方法。

function dedupe(array) {
  return Array.from(new Set(array));
}

dedupe([1, 1, 2, 3]) // [1, 2, 3]

1.3 遍历操作

Set 结构的实例有四个遍历方法,可以用于遍历成员。

Set.prototype.keys():返回键名的遍历器
Set.prototype.values():返回键值的遍历器
Set.prototype.entries():返回键值对的遍历器
Set.prototype.forEach():使用回调函数遍历每个成员

需要特别指出的是,Set的遍历顺序就是 插入顺序。这个特性有时非常有用,比如使用 Set 保存一个回调函数列表,调用时就能保证按照添加顺序调用。

- values(),entries()

keys方法、values方法、entries方法返回的都是遍历器对象(详见《Iterator 对象》一章)。由于 Set 结构没有键名,只有键值(或者说键名和键值是同一个值),所以keys方法和values方法的行为完全一致。

let set = new Set(['red', 'green', 'blue']);

for (let item of set.keys()) {
  console.log(item);
}
// red
// green
// blue

for (let item of set.values()) {
  console.log(item);
}
// red
// green
// blue

for (let item of set.entries()) {
  console.log(item);
}
// ["red", "red"]
// ["green", "green"]
// ["blue", "blue"]

☆☆☆ Set 默认遍历器 === values()

Set 结构的实例默认可遍历,它的默认遍历器生成函数就是它的values方法。

Set.prototype[Symbol.iterator] === Set.prototype.values  // 查看成员方法!
// true

这意味着,可以省略values方法,直接用for...of循环遍历 Set。

let set = new Set(['red', 'green', 'blue']);

for (let x of set) {
  console.log(x);
}
// red
// green
// blue

- values(),entries()

你可能感兴趣的:(ES6,ecmascript)