ES6 新增了兩個數據結構,一個是 set,另一個是 map。前端
在《你不知道的 JavaScript(下卷)》中是這麼定義的:set 是一個值的集合,其中的值惟一(重複會被忽略)。 它相似於數組,可是每一個成員的值是惟一的。 set 是一個構造函數,能夠經過 new 來建立一個 set 實例。git
let set = new Set([1, 2, 3, 1, 4]);
console.log(set); // {1, 2, 3, 4} 複製代碼
上述能夠看出,new Set 會自動過濾掉重複值,並返回一個集合,咱們能夠利用這個特性,來寫出簡單的數組去重。es6
let set = new Set([1, 2, 3, 1, 4]);
let arr = [...set]; console.log(arr); // [1, 2, 3, 4] 複製代碼
set 經過 add()來增長成員,將新值放在集合尾部,若是新值跟原集合中的成員重複的話,會被自動過濾掉。github
let set = new Set([1, 2, 3, 1, 4]);
set.add(1); console.log(set); // {1, 2, 3, 4} let set = new Set([1, 2, 3, 1, 4]); set.add(0); console.log(set); // {1, 2, 3, 4, 0} 複製代碼
Set 的構造函數能夠接受一個具備 Iterable 接口的其餘數據結構做爲參數。編程
// 例1
let set = new Set("string"); console.log(set); // {"s", "t", "r", "i", "n", "g"} // 例2 function foo(a, b) { let set = new Set(arguments); console.log(set); // {1, 2} } foo(1, 2); 複製代碼
set 的實例屬性有兩個:數組
實例方法分爲兩類,一個是操做類,一個是遍歷類。 操做類:markdown
// 例1
let set = new Set([1, 2, 3]); set.clear(); console.log(set); // {} // 例2 let set = new Set([1, 2, 3]); let result = set.delete(1); console.log(result); // true console.log(set); // {2, 3} // 例3 let set = new Set([1, 2, 3]); let result = set.has(1); console.log(result); // true 複製代碼
遍歷類:數據結構
// 例1
let set = new Set([1, 2, 3]); let keys = set.keys(); console.log(keys); // {1, 2, 3} // 例2 let set = new Set([1, 2, 3]); let values = set.values(); console.log(values); // {1, 2, 3} 複製代碼
上述兩個例子,由於 Sset 沒有鍵名,只有值,因此 keys 和 values 的結果是同樣的。 Set 一樣擁有 forEach 方法:函數
let set = new Set([1, 2, 3]);
set.forEach((key, value) => { console.log(`${key}:${value}`); }); // 1:1 // 2:2 // 3:3 複製代碼
跟數組的 forEach 方法不一樣的是,它的回調函數的參數是其 key 值和 value 值,可是 Set 沒有 key 值,因此其 key 和 value 相等。forEach 的第二個參數是 this 值,其綁定的是回調函數中的 this 值。 最後一個實例方法是 entries:oop
let set = new Set([1, 2, 3]);
let entries = set.entries(); console.log(entries); // { // [1, 1], // [2, 2], // [3, 3] // } 複製代碼
enteries()返回一個遍歷器,返回一個鍵值對,但其鍵值對均相等。
WeakSet 也是一個構造函數,與 Set 一直,WeakSet 和 Set 二者類似但不一樣,不一樣點主要有兩個:
let weakSet = new WeakSet([1, 2]); // Uncaught TypeError: Invalid value used in weak set
let weakSet = new WeakSet([ [1, 2], [3, 4], ]); // {[1, 2], [3, 4]} 複製代碼
WeakSet 有三個實例方法:
var ws = new WeakSet();
var foo = {}; var bar = {}; ws.add(foo); ws.add(bar); ws.has(foo); // true ws.has(bar); // true ws.delete(foo); // 從set中刪除 foo 對象 ws.has(foo); // false, foo 對象已經被刪除了 ws.has(bar); // true, bar 依然存在 複製代碼
在 JS 中對象是建立無序鍵 / 值對數據結構 [ 也稱爲 映射(map)] 的主要機制。可是,對象做爲映射的主要缺點是不能使用非字符串值做爲鍵。因此在 ES6 提出一個新的數據結構,Map。 Map 和對象很相似,都是鍵值對的形式,可是 Map 的鍵能夠是任意類型(對象或者原始值),NaN 也能夠做爲鍵,再也不只侷限於字符串。 任何具備 Iterator 接口、且每一個成員都是一個雙元素的數組的數據結構均可以看成 Map 構造函數的參數。
let map = new Map([[{a:1},3]]);
console.log(map);//{ { a:1 },3} 複製代碼
在 MDN 中清晰的列出了 Map 和 Object 間的不一樣:
Map | Object | |
---|---|---|
意外的鍵 | Map 默認狀況不包含任何鍵。只包含顯式插入的鍵。 | 一個 Object 有一個原型, 原型鏈上的鍵名有可能和你本身在對象上的設置的鍵名產生衝突。注意: 雖然 ES5 開始能夠用 Object.create(null) 來建立一個沒有原型的對象,可是這種用法不太常見。 |
鍵的類型 | 一個 Map 的鍵能夠是任意值,包括函數、對象或任意基本類型。 | 一個 Object 的鍵必須是一個 String 或是 Symbol。 |
鍵的順序 | Map 中的 key 是有序的。所以,當迭代的時候,一個 Map 對象以插入的順序返回鍵值。 | 一個 Object 的鍵是無序的。注意:自 ECMAScript 2015 規範以來,對象確實保留了字符串和 Symbol 鍵的建立順序; 所以,在只有字符串鍵的對象上進行迭代將按插入順序產生鍵。 |
Size | Map 的鍵值對個數能夠輕易地經過 size 屬性獲取 | Object 的鍵值對個數只能手動計算 |
迭代 | Map 是 iterable 的,因此能夠直接被迭代。 | 迭代一個 Object 須要以某種方式獲取它的鍵而後才能迭代。 |
性能 | 在頻繁增刪鍵值對的場景下表現更好。 | 在頻繁添加和刪除鍵值對的場景下未做出優化。 |
Map 的實例屬性有兩個:
實例方法分爲兩類,一個是操做類,一個是遍歷類。 操做類:
let map = new Map();
let key = {name:"Jack"}; let value = "啥"; map.set(key, value); //{ { name:'Jack' }:'啥'} map.get(key); // '啥' map.get(a); // undefined map.has(key); // true map.delete(key); // true map.clear(); // {} 複製代碼
遍歷類:
let map = new Map([
["name","Jack"], [{age:25},"啥"], ]); map.entries(); // MapIterator {"name"=>"Jack",{age:25}=>"啥"} for (let item of map.entries()) { console.log(item); // ["name","Jack"] [{age:25},"啥"] } map.forEach((value, key, map) => { console.log(value, key, map); // Jack name Map(2) {"name"=>"Jack",{age: 25}=>"啥"} //"啥"{age:25} Map(2) {"name" =>"Jack",{age:25}=>"啥"} }); map.keys(); // MapIterator {"name",{age:25}} for (let item of map.keys()) { console.log(item); // "name" {age:25} } map.values(); // MapIterator {"Jack","啥"} for (let item of map.values()) { console.log(item); // "Jack" "啥" } 複製代碼
ES6 給出一個最簡單的方式,擴展運算符。
let map = new Map([
["name", "Jack"], [{ age: 25 }, "啥"], ]); console.log([...map]); // [['name', 'Jack'], [{age: 25}, '啥']] 複製代碼
除此以外還能夠用 Array.from()。
let map = new Map([
["name", "Jack"], [{ age: 25 }, "啥"], ]); console.log(Array.from(map)); // [['name', 'Jack'], [{age: 25}, '啥']] 複製代碼
若是全部 Map 的鍵都是字符串,它能夠無損地轉爲對象。
function strMapToObj(strMap) {
let obj = Object.create(null); for (let [k, v] of strMap) { obj[k] = v; } return obj; } const myMap = new Map().set("yes", true).set("no", false); strMapToObj(myMap); // { yes: true, no: false } 複製代碼
Map 轉 JSON 有兩個形式,第一種是鍵名都是字符串的類型的,第二種是鍵名中包含非字符串類型。 第一種的話直接將 map 轉爲對象,而後再用 JSON.Stringfy()進行轉換。 第二種的話能夠轉成數組 JSON。
function mapToArrayJson(map) {
return JSON.stringify([...map]); } let myMap = new Map().set(true, 7).set({ foo: 3 }, ["abc"]); mapToArrayJson(myMap); // '[[true,7],[{"foo":3},["abc"]]]' 複製代碼
WeakMap 和 WeakSet 有些相似,都是弱引用,而且成員值的鍵必須是對象。
let weakMap = new WeakMap([[1,3]]); // Uncaught TypeError: Invalid value used as weak map key
let weakMap = new WeakMap([[{name:"Jack"},3]]); // {{name:'Jack'}:3} 複製代碼
WeakMap 有四個實例方法:
let weakMap = new WeakMap();
let obj = {name: 'Jack'}; weakMap.set(obj, 3); // {{name:'Jack'}: 3} weakMap.get(obj); // 3 weakMap.has(obj); // true weakMap.delete(obj); // true 複製代碼
Map 的應用最直接的就是策略模式。舉個例子若是一個公司的年終獎爲一等獎是電腦,二等獎是手機,三等獎是步步高點讀機,四等獎是礦泉水。那通常會這麼寫:
function getAnnualBonus(level) {
if (level === "一等獎") { return "電腦"; } else if (level === "二等獎") { return "手機"; } else if (level === "三等獎") { return "步步高點讀機"; } else if (level === "四等獎") { return "礦泉水"; } } getAnnualBonus("一等獎"); // 或者 function getAnnualBonus(level) { switch (level) { case "一等獎": return "電腦"; case "二等獎": return "手機"; case "三等獎": return "步步高點讀機"; case "四等獎": return "礦泉水"; default: break; } } getAnnualBonus("一等獎"); 複製代碼
上述兩種寫法,若是條件愈來愈多的話,那寫的 if...else...和 case 愈來愈長,代碼至關的臃腫。 運用 Map 來寫策略模式,簡潔明瞭:
let annualBonus = new Map([
['一等獎', '電腦'], ['二等獎', '手機'], ['三等獎', '步步高點讀機'], ['四等獎', '礦泉水'] ]); function getAnnualBonus(level) { return annualBonus.get(level); } getAnnualBonus("一等獎"); 複製代碼
相關文章:
以爲還能夠的,麻煩走的時候能給點個贊,讓更多人能看到這篇文章,你們一塊兒學習和探討!不點讚的都是耍流氓😂
還能夠關注個人博客但願能給個人github上點個Start,小夥伴們必定會發現一個問題,個人全部用戶名幾乎都與番茄有關,由於我真的很喜歡吃番茄❤️!!!
想跟車不迷路的小夥還但願能夠關注公衆號 前端老番茄 或者掃一掃下面的二維碼👇👇👇。
我是一個編程界的小學生,您的鼓勵是我不斷前進的動力,😄但願能一塊兒加油前進。
本文使用 mdnice 排版