其實數組也是集合, 只不過數組的索引是數值類型.當想用非數值類型做爲索引時, 數組就沒法知足須要了.數組
而 Map 集合能夠保存多個鍵-值對(key-value), Set 集合能夠保存多個元素.函數
對Map 和 Set 通常不會逐一遍歷其中的元素. Map 通常用來存儲須要頻繁取用的數據, Set 通常用來判斷某個值是否存在其中.post
在ES 5 中,沒有 Set和Map集合, 通常使用對象來模擬這兩種集合, 對象的屬性做爲鍵(key), 以屬性值做爲值(value), 即以 property: property-value
來模擬 key-value
的形式. 具體實現以下:ui
模擬 Map 的鍵值對集合:spa
// 建立一個 Map 對象
var map = Object.create(null);
// 添加屬性和屬性值, 即 添加 key 和 value
map.key1 = 'value 1';
map.key2 = {};
// 取得 key 對應的 value
console.log(map.key1); // "value 1"
console.log(map.key2); // "Object {}"
複製代碼
模擬 Set :3d
// 建立一個 Set 對象
var set = Object.create(null);
// 添加屬性和屬性值, 即 添加 key 並令其值爲 true, 即表示這個key存在於集合中
set.key = true;
// 判斷 key 是否存在, 而後進行下一步的操做
if(set.key) { ... }
複製代碼
下面正式來討論這兩種集合的特色code
Map 中存儲的是 key-value 形式的鍵值對, 其中的 key 和 value 能夠是任何類型的, 即對象也能夠做爲 key . 這比用對象來模擬的方式就靈活了不少對象
// 建立一個空的 Map
let map = new Map();
複製代碼
// 用數組來建立一個 非空的 Map
let array = [ // 定義一個二維數組, 數組中的每子數組都有兩個元素
['key1' , 'value 1'], // key 是 字符串 "key1", value 是字符串 "value 1"
[{} , 10086] , // key 是個對象, value 是數值 10086
[ 5, {} ] // key 是個數值類型, value 是對象
];
let map = new Map(array); // 將數組傳入 Map 構造函數中
複製代碼
set(key, value)
: 向其中加入一個鍵值對get(key)
: 若不存在 key 則返回 undefined
has(key)
:返回布爾值delete(key)
: 刪除成功則返回 true, 若key不存在或者刪除失敗會返回 falseclear()
: 將所有元素清除和數組的 forEach 方法相似, 回調函數中都包含3個參數 值, 鍵, 和 調用這個方法的 Map 集合自己索引
map.forEach(function(value, key, ownerMap){
console.log(key, value); // 每對鍵和值
console.log(ownerMap === map); // true
});
複製代碼
Set 和 Map 最大的區別是隻有鍵 key 而沒有 value, 因此通常用來判斷某個元素(key)是否存在於其中.生命週期
既能夠建立一個空 set 也能夠用數組來初始化一個非空的set. 和 Map 不一樣的是, 數組是一維數組, 每一個元素都會成爲 set 的鍵.例如:
// 建立一個數組
let array = [1, 'str']; // 一維數組
// 用數組來初始化 set
let set = new Set(array);
複製代碼
add(key)
: 往set添加一個元素, 若是傳入多個參數, 則只會把第一個加入進去let set = new Set();
set.add(1, 2, 3);
console.log(set.has(1), set.has(2), set.has(3)); // true false false 能夠看到只有第一個參數被加入進了 set
複製代碼
has(key)
delete(key)
clear()
和 Map 的 forEach 方法類似, 回調函數的參數也是3個 (value, key, ownerSet). 按道理來講由於 set 中只有 key 沒有 value, 那麼回調函數中不該該存在 value 這個參數呀, 那爲何還會有 value 這個參數呢?多是由於 另外兩種集合 ( 數組和 Map ) 的 forEach 方法的回調函數的參數都是這三個, 若是對於 Set 不是這樣, 那麼這三種集合的 forEach 函數就會丟失了一致性. 這個理由......
那麼既然沒有 value , 那麼這個value的值是什麼呢?答案是: value的值 === key的值 ,也就是說 在這裏 value 就是 key, 只不過在字面上寫成了 value 而已. 爲了保持一致性嘛
下面這段代碼能夠驗證這個說法.
set.forEach(function(value, key, ownerSet){
console.log(value === key, set === ownerSet); // true true
});
複製代碼
這兩個集合比以前的兩個集合在名字以前都加上了 Weak
, 這個 Weak
能夠直譯成弱
, 這個弱指的是弱引用, 那麼前面不帶Weak的 Set 和 Map 就是不弱, 也就是強了. 這個強指的是強引用, 再具體一點就是對於 key 位置的對象的強引用. 下面會詳細討論.
add, has, delete
方法可用; WeakMap 只有 set, has, get, delete
方法可用.強弱版本對於 key 是對象時的引用機制以下:
將對象設置爲 key 時, 就在集合中保存了這個對象的引用. 當這個對象沒有其餘引用了的時候, ( 即只有集合還引用着這個對象的時候 ), 弱類型的集合會放棄對這個對象的引用, 把這個對象從集合裏移除, 不讓它繼續存在於集合中了, 這樣一來, 這個對象就會被垃圾回收了, 很有些「趕盡殺絕」的意思; 可是強類型的集合還會一直保存着對這個對象的引用, 就把它一直放在集合裏.這就是 [WeakSet 和 WeakMap] 與 [Set 和 Map] 的根本區別.
要注意的是這個機制只做用於 key , 而 value 位置綁定的對象不管是否還存在別的引用, WeakMap 都不會放棄這個對象. 只有這個位置的 key 綁定的對象沒有其餘引用時, 纔會把 key 和 value 都放棄. 是否放棄的決定權在於 key 位置.
若版本集合能夠用在須要生命週期管理的地方,例如保存對一個 DOM 對象的引用, 若是一個 DOM 對象使用完畢, 沒有其餘的引用了, 那麼它應該被 垃圾回收,以避免產生內存泄漏,那麼弱版本的集合就最適合用來保存這樣的對象了。