ES6的Set和Map数据结构

ES6的Set和Map数据结构

最近在写业务代码的时候经常碰到数组或者对象去重问题,常用的方法当然就是遍历,然后,我自己搜了一下有没有更简洁的做法,发现了es6的Set和Map数据结构也可以用于去重,最主要的是省了好多代码。

那么Set和Map到底是何方神圣呢,刚开始看的时候越看越懵逼,因为我不明白实现这样的数据结构,目的是什么,意义又是什么呢?

Set和Map主要的应用场景在于数组去重和数据存储,幸运的是在读了关于数据结构和算法之类的书籍后,恍然大悟的发现

原来Set是一种叫做集合的数据结构,Map是一种叫做字典的数据结构

那么下面就跟随我一起去了解下这两种数据结构,最后来亲手实现的一个ES6中的Set和Map吧

集合

  1. 集合是由一组无序且唯一(即不能重复)的项组成的,可以想象成集合是一个既没有重复元素,也没有顺序概念的数组
  2. ES6提供了新的数据结构Set。它类似于数组,但是成员的值都是唯一的,没有重复的值
  3. Set 本身是一个构造函数,用来生成 Set数据结构

    这里说的Set其实就是我们所要讲到的集合,先来看下基础用法

1
2
3
4
5
6
7
8
9
10
11
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
}

// 去除数组的重复成员
let array = [1,2,1,4,5,3];
[...new Set(array)] // [1, 2, 4, 5, 3]

Set实例的属性和方法

Set的属性:

  • size:返回集合所包含元素的数量

Set的方法:
操作方法——

  • add(value):向集合添加一个新的项
  • delete(value):从集合中移除一个值
  • has(value):如果值在集合中存在,返回true,否则false
  • clear(): 移除集合里所有的项

遍历方法——

  • keys():返回一个包含集合中所有键的数组
  • values():返回一个包含集合中所有值的数组
  • entries:返回一个包含集合中所有键值对的数组(感觉没什么用就不实现了)
  • forEach():用于对集合成员执行某种操作,没有返回值

Set完整实现
在此,先给大家贴一下完整的实现代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
function Set(arr = []) {
let items = {};
this.size = 0;

// has方法
this.has = function (val) {
return items.hasOwnProperty(val);
};

// add方法
this.add = function (val) {
// 如果没有存在items里面就可以直接写入
if (!this.has(val)) {
items[val] = val;
this.size++;
return true;
}
return false;
};
arr.forEach((val, i) => {
this.add(val);
});

// delete方法
this.delete = function (val) {
if (this.has(val)) {
delete items[val]; // 将items对象上的属性删掉
this.size--;
return true;
}
return false;
};

// clear方法
this.clear = function () {
items = {};
this.size = 0;
};

// keys方法
this.keys = function () {
return Object.keys(items);
};

// values方法
this.values = function () {
return Object.values(items);
}
// forEach方法
this.forEach = function (fn, context = this) {
for (let i = 0; i < this.size; i++) {
let item = Object.keys(items)[i];
fn.call(context, item, item, items);
}
}

// 并集
this.union = function (other) {
let union = new Set();
let values = this.values();

for (let i = 0; i < values.length; i++) {
union.add(values[i]);
}
values = other.values(); // 将values重新赋值为新的集合
for (let i = 0; i < values.length; i++) {
union.add(values[i]);
}

return union;
};

// 交集
this.intersect = function (other) {
let intersect = new Set();
let values = this.values();
for (let i = 0; i < values.length; i++) {
if (other.has(values[i])) {
intersect.add(values[i]);
}
}
return intersect;
};

// 差集
this.difference = function (other) {
let difference = new Set();
let values = this.values();
for (let i = 0; i < values.length; i++) {
if (!other.has(values[i])) {
difference.add(values[i]);
}
}
return difference;
};

// 子集
this.subset = function(other) {
if (this.size > other.size) {
return false;
} else {
let values = this.values();
for (let i = 0; i < values.length; i++) {
console.log(values[i])
console.log(other.values())
if (!other.has(values[i])) {
return false;
}
}
return true;
}
};
}

module.exports = Set;

写了辣么多一起来测试一下吧

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
const Set = require('./Set.js');
let set = new Set([2, 1, 3]);
console.log(set.keys()); // [ '1', '2', '3' ]
console.log(set.values()); // [ 1, 2, 3 ]
console.log(set.size); // 3
set.delete(1);
console.log(set.values()); // [ 2, 3 ]
set.clear();
console.log(set.size); // 0

// 并集
let a = [1, 2, 3];
let b = new Set([4, 3, 2]);
let union = new Set(a).union(b).values();
console.log(union); // [ 1, 2, 3, 4 ]

// 交集
let c = new Set([4, 3, 2]);
let intersect = new Set([1,2,3]).intersect(c).values();
console.log(intersect); // [ 2, 3 ]

// 差集
let d = new Set([4, 3, 2]);
let difference = new Set([1,2,3]).difference(d).values();
// [1,2,3]和[4,3,2]的差集是1
console.log(difference); // [ 1 ]

字典

在数据结构还有一种结构叫做字典,它就是实现基于ES6中的Map类的结构

那么集合又和字典有什么区别呢:

  • 共同点:集合、字典可以存储不重复的值
  • 不同点:集合是以[值,值]的形式存储元素,字典是以[键,值]的形式存储
  • 所以这一下让我们明白了,Map其实的主要用途也是用于存储数据的,相比于Object只提供“字符串—值”的对应,Map提供了“值—值”的对应。也就是说如果你需要“键值对”的数据结构,Map比Object更合适

下面来看一下基本使用:

1
2
3
4
5
6
7
8
const m = new Map();
const o = {p: 'Hello World'};
m.set(o, 'content')
m.get(o) // "content"

m.has(o) // true
m.delete(o) // true
m.has(o) // false

以上是Map的基本使用,还有更多有用的方法稍后会随着实现的深入分别展示

Map的属性和方法

属性:

  • size:返回字典所包含的元素个数

操作方法:

  • set(key, val): 向字典中添加新元素
  • get(key):通过键值查找特定的数值并返回
  • has(key):如果键存在字典中返回true,否则false
  • delete(key): 通过键值从字典中移除对应的数据
  • clear():将这个字典中的所有元素删除

遍历方法:

  • keys():将字典中包含的所有键名以数组形式返回
  • values():将字典中包含的所有数值以数组形式返回
  • forEach():遍历字典的所有成员

Map完整实现

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
function Map() {
let items = {};
this.size = 0;

// 操作方法
// has方法
this.has = function(val) {
return items.hasOwnProperty(val);
};

// set(key, val)方法
this.set = function(key, val) {
items[key] = val;
this.size++;
};

// get(key)方法
this.get = function(key) {
return this.has(key) ? items[key] : undefined;
};

// delete(key)方法
this.delete = function(key) {
if (this.has(key)) {
delete items[key];
this.size--;
return true;
}
return false;
};

// clear()方法
this.clear = function() {
items = {};
this.size = 0;
};

// 遍历方法
// keys()方法
this.keys = function() {
return Object.keys(items);
};

// values()方法
this.values = function() {
return Object.values(items);
};

// forEach(fn, context)方法
this.forEach = function(fn, context = this) {
for (let i = 0; i < this.size; i++) {
let key = Object.keys(items)[i];
let value = Object.values(items)[i];
fn.call(context, value, key, items);
}
};
}

module.exports = Map;

再来看看下面的测试:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
// map.js
// 使用Map类
const Map = require('./Map.js');
let m = new Map();
m.set('Jay', 'Jay的Chou');
m.set(true, '真的');
console.log(m.has('Chou')); // false
console.log(m.size); // 2
console.log(m.keys()); // [ 'Jay', 'true' ]
console.log(m.values()); // [ 'Jay的Chou', '真的' ]
console.log(m.get('jay')); // undefined

m.delete(true);
console.log(m.keys()); // [ 'Jay' ]
console.log(m.values()); // [ 'Jay的Chou' ]
# es6, 前端
Your browser is out-of-date!

Update your browser to view this website correctly. Update my browser now

×