|
|
& m% g$ F2 M) ^' L% @; X
<p><span style="background-color: rgba(194, 224, 244, 1)"><span style="background-color: rgba(52, 73, 94, 1)"><span style="color: rgba(255, 255, 255, 1)">我认为前端生态很大,犹如一片汪洋大海,很难短时间内窥其全貌,在这里我们不谈其他,只聊聊</span> <span style="color: rgba(255, 255, 255, 1)">Set</span></span></span></p>8 C, @* ~$ ?5 V6 G, c2 U% E
<p><span style="background-color: rgba(52, 73, 94, 1)"><span style="color: rgba(255, 255, 255, 1)">Set</span></span>是 es6 新提出的一个对象,也是一种数据结构,为什么es6要提出这样一个新对象呢,无非就是丰富js的对象类型,在遇到具体的业务场景需要一个比较适合的恰当的数据结构来保存和操作数据,接下来就让我们更深刻的认识下<span style="background-color: rgba(52, 73, 94, 1)"><span style="color: rgba(255, 255, 255, 1)">Set</span></span>的一些常用的方法和应用场景</p>* U. j' p! `% @. N" v1 R- _8 K% L
<p>首先 <span style="background-color: rgba(52, 73, 94, 1)"><span style="color: rgba(255, 255, 255, 1)">Set</span></span>的一个重要特性就是集合中是不允许添加重复元素的,如何判断重复,如果是基本类型的话根据===操作符,如果是引用类型的话是根据对象的指针是否指向同一个引用对象,特别有意思的是NaN,<span style="background-color: rgba(52, 73, 94, 1)"><span style="color: rgba(255, 255, 255, 1)">Set</span></span>是把它作为相同对待的,尽管NaN本身是不相等的,所以<span style="background-color: rgba(52, 73, 94, 1)"><span style="color: rgba(255, 255, 255, 1)">Set</span></span>中的值都是唯一的</p>
) `2 P- D8 C1 J2 X! \8 ^+ o<p><span style="background-color: rgba(52, 73, 94, 1)"><span style="color: rgba(255, 255, 255, 1)">Set</span></span>是一个构造函数,所以我们使用<span style="background-color: rgba(52, 73, 94, 1)"><span style="color: rgba(255, 255, 255, 1)">Set</span></span>必须使用new关键字</p>$ ^" ]5 y* j! |4 K
<p>两种方法创建</p>- k( p, H- e& k5 U9 E5 K a
<ul>
7 n. A8 y0 y3 [% Y<li> 直接创建一个空的集合
( ^6 C4 V% ^! @0 L0 P" P: s<pre class="language-javascript"><code>const set = new Set()</code></pre>
3 m, e) K) F, D) R9 L4 m0 f D' i* ?</li>8 L0 k1 V& B) N9 o( J
<li>传入一个数组或者具有iterable接口的其他数据结构 7 k: J6 G$ S* m3 b5 W/ R' [. u
<pre class="language-javascript"><code>const set = new Set([1,2,3,4,5])' m) r% e8 L# k4 e; C) V' W4 u
const set = new Set('我是字符串,我具有iterable接口哦')</code></pre>
. X H0 ?1 o" T4 A</li>+ Y0 R; Y7 _ k5 x- E8 P+ o
</ul>% _! P, ?3 `' K( B3 V6 P% j; w
<p><strong>Set 的实例属性和方法</strong></p>! k$ X, i4 n* x, U
<ol style="list-style-type: upper-roman">9 H, U9 @( s/ Y8 j
<li>操作方法( |' `+ L% v9 Q6 P- o P: c2 P2 g
<ol>
; H, s- Q+ @' b- A5 {! p8 J. H! b<li>添加
5 O, [; ]& ~, h# F- g1 M<pre class="language-javascript"><code>const s = new Set()& V( \ i2 B1 b" a* f4 G& |
// add 方法返回Set实例本身,所以可以执行链式操作
0 x. {/ c$ D- d+ Jconst ret = s.add(1).add('one').add({1:'one'})</code></pre>1 q) m' ?0 {3 O; s9 J/ i
</li>
8 i1 B( t) _2 K: `8 t7 Y6 U<li>删除
- Q+ v& t1 F/ v1 u4 ]<pre class="language-javascript"><code>const s = new Set([1,2,3,4,5])
' h- f8 ?; i; }3 r+ M: n4 v; d |// delete 方法返回被删除元素是否删除成功4 w$ w7 [8 R5 F+ L: W
const flag = s.delete(1) // true1 U9 B2 Q: z' x) Y( D K) B* E! L! W9 c
const flag = s.delete('2') // false</code></pre>
+ G' _# O6 E/ G( G$ u5 [</li> h3 A$ f8 _/ m3 Z: N
<li>查找
; G K4 u/ K# K+ w( V$ P! s<pre class="language-javascript"><code>const s = new Set([1,2,4,5,6,7])+ s1 I4 r. N8 @, W" F8 j
const flag = s.has(2)</code></pre>
- ?9 O: u0 ~0 f- m</li>
; ?6 R! K, u5 s, p* b6 a<li>清空" |6 I7 g: S( C4 b* D
<pre class="language-javascript"><code>const s = new Set([12,324,1])
" O5 S3 X2 {1 P# I0 \/ |3 s//clear 方法没有返回值,返回undefined
f# A1 Y1 g- Us.clear()</code></pre> `# ~$ q8 f; \8 z Z2 g. s
</li>( _. f6 r/ X" c" P0 w7 o
<li>两个实例属性<br>
* c* y: l) L( C: B3 p<pre class="language-javascript"><code>const s = new Set()' Q: D- y" P w' m/ P- a8 G0 x+ }# h
// 实例还有两个属性8 X+ ~; J' j0 S5 I2 c
s.size //返回当前集合中元素个数
- V8 `. E7 U( X2 V! g) is.constructor // 返回实例构造器,也就是Set</code></pre>
: S; {/ M4 Q5 ~( h4 r# L9 S2 F$ a<p> </p>
4 P+ m9 O7 D- }0 Z( o& B7 |7 ]: f</li>: X( V: n/ D, L3 n* I) I7 b
</ol>
& _8 f7 ^) A' ]( i/ c. D</li>
- M, k" ~2 f2 { X6 F! x, e<li>遍历方法</li>
. g$ U, l: N/ O, k; k, I</ol>- {3 `9 u: \3 Z( ^
<ol>. |, R8 c( f! u4 x2 o; R
<li style="list-style-type: none"> |! M4 j% y- M Y) E, l
<ol>
7 n: s \3 v, }/ y ~1 v<li>遍历键名$ V2 ` Y. c; N( e# R
<pre class="language-javascript"><code>const s = new Set(['javascript','html','css']). z9 }9 P4 }9 K6 B# f+ j1 x
& z7 I* h+ t7 W; [$ o$ Ofor(let key of s.keys()){1 z) ^: T% U2 O7 l+ S# T0 f0 h
console.log(key)
2 k% {& R6 v' A, Q" v1 G; I! E* d) t}
! [# L t- X" A//javascript
# R u; q5 L; G' y//html
5 E8 ?- _) W1 M+ I; ]0 x//css- U$ K, Y6 \# o0 u8 w9 c$ p
//遍历顺序就是插入顺序,利用这个特性可以储存一些需要按顺序调用的函数</code></pre>
K* A9 s; E" ~/ \9 X' H</li>; T' P2 {, K6 Z7 `% x
<li>遍历键值
+ }% u( Y# e# M<pre class="language-javascript"><code>const s = new Set(['javascript','html','css'])% ?# w- T2 n2 e# d8 R* a6 W+ ]+ ~
( b2 w/ ?, b4 E9 C
for(let value of s.values()){
% x3 r& u* x" H0 d- {6 I. O f* F console.log(value)
" }2 j( X; o! P" @4 p; Z}
' B' [# R+ w# c//Set不存在键名,只有键值,也可以认为键名和键值是同一个,所以keys和values返回的值是一样的</code></pre>/ i! g9 N, ~& v, S4 Y" ~) ~
</li>
4 u# [9 u. b; _# i2 P& ]% ~; a: j9 l<li>遍历键值对
% ~" e3 D) m2 ~- V, G9 r1 y4 m9 x<pre class="language-javascript"><code>const s = new Set(['javascript','html','css'])
* ?# V0 a( u& ]# K7 q4 G8 N0 _5 _' X0 |7 f }% w" e, I* v
for(let entry of s.entries()){4 Y" ^: N4 }! i% W \
console.log(entry)
! A3 Y D& L: A5 L9 J ]' m}9 n$ M7 e* x! R; k( G2 u4 P/ h
//['javascript', 'javascript']
- W- o/ v9 t7 Z& X: w//['html', 'html']! R' m4 }5 H' w8 D
//['css', 'css']
7 M4 Z- D) x) n1 L9 b9 [: ~1 Q//遍历的每一对都是一个包括键名和键值的数组</code></pre>
0 I+ H, L+ a. ?) X- X( d4 U</li>
/ m% y3 @) x, j1 u) z- t<li>forEach 使用回调函数遍历每一个元素- T8 e6 g" J, `# k' B7 H. e7 \. A) t
<pre class="language-javascript"><code>const s = new Set(['javascript','html','css'])
( ?5 n6 d9 ]- T0 P( w" l' _) gs.forEach(function(value,key,s) {
+ T3 ^, Q8 e' _% h1 ]1 U3 Z" K // 回调函数接受三个参数,键值,键名,set本身8 e+ @" \% z7 r# n2 @1 C) Y! h5 k
console.log(`键值:${value};键名${key};集合大小${s.size};${this.thisName}`), E* j* J" @ z
},{thisName:'改变回调函数this'})
8 p/ R+ `( _- l1 `// forEach函数还接受第二个参数,可以绑定处理函数的this</code></pre>
5 P7 ?9 Z" |" U+ o n- c+ H</li>: g3 ^7 a! ^0 ~1 I4 Z3 b* w( V
<li>Set实例默认是可以迭代的,因为它的遍历器生成函数其实调用的就是values方法,这意味着我们可以直接省略values()方法直接遍历
/ g( {: L G2 V( J) t8 }8 |$ b" ]<pre class="language-javascript"><code>const s = new Set(['javascript','html','css'])
7 G. _0 b3 ]( \9 s! c; F& ~: v" A
1 M. r% H! N8 O; A, t3 R/ R/ zfor(let value of s){' h* y/ {2 r9 M% R
console.log(value)# t% [! U" u8 C' d% j) n- a
}</code></pre>
: n& B# f Z( p' @8 y<p> </p>3 j6 O5 g1 [8 n/ i6 v \ ]
</li>( _+ v; N& G! T$ D
</ol>
0 n8 b) ?, Q& g8 c% S% G</li>
! G8 H) W& y3 U5 R" }* j3 B6 G</ol> |3 F1 G0 Q5 S$ ?- k) y F% l
<p><strong>应用</strong></p>; R) l4 @' s3 f
<p>1、我们首先可以结合扩展操作符(...)给数组去重</p>
4 S) @ I9 L# Y5 t6 C. f. L4 d+ Q<pre class="language-javascript"><code>const unique = [...new Set([1,2,3,4,5,1,21,23,5])]
, \% D. r; x1 R$ p//[1, 2, 3, 4, 5, 21, 23]</code></pre>
3 C8 ^9 r- C$ X4 s* N<p>2、实现并集,交集,差集</p>
* e8 h2 |) j+ _3 F<pre class="language-javascript"><code>const s1 = new Set([1,2,3,4])- [$ T3 j- F7 P, p
const s2 = new Set([2,3,4,5])
# T {' u7 I. B, a
/ n/ A9 J# h8 s//并集 Set(5) {1, 2, 3, 4, 5}2 G8 S; o3 H! C0 H4 B4 q& b
const union = new Set([...s1,...s2])
# M! @ d% [" f5 {' p; N. J
8 H* z* V k, j# ^8 Y% e, l//交集 Set(3) {2, 3, 4}/ M# R! D: a/ A" w; Q- A o" r
const intersect = new Set([...s1].filter(v => s2.has(v)))$ z9 o, u- y* r
( j. O. q/ J3 X) n" Z3 k( |
//差集 i9 g& \' A8 y- d
const difference = new Set([...new Set([...s1].filter(v => !s2.has(v))),...new Set([...s2].filter(v => !s1.has(v)))])</code></pre>% P. V! Q, w' L$ z1 d
<p> </p>( _1 }; a2 a i
<p>Set 基本的用法就先讲到这里,有不对的地方欢迎大家指正</p>
# Q( F# a5 p2 ^6 p8 ~" q/ o& S- T& H |
|