溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

如何使用map代替純JS對象

發(fā)布時間:2020-12-04 11:37:40 來源:億速云 閱讀:158 作者:小新 欄目:web開發(fā)

這篇文章主要介紹了如何使用map代替純JS對象,具有一定借鑒價值,需要的朋友可以參考下。希望大家閱讀完這篇文章后大有收獲。下面讓小編帶著大家一起了解一下。

JavaScript 普通對象 {key: 'value'} 可用于保存結(jié)構(gòu)化數(shù)據(jù)。

但是我發(fā)現(xiàn)很煩人的一件事:對象的鍵必須是字符串(或很少使用的符號)。

如果用數(shù)字作鍵會怎樣?在這種情況下沒有錯誤:

const names = {
  1: 'One',
  2: 'Two',
};

Object.keys(names); // => ['1', '2']

JavaScript 只是將對象的鍵隱式轉(zhuǎn)換為字符串。這是一件棘手的事,因為你失去了類型的一致性。

在本文中,我將介紹 ES2015 中提供的 JavaScript Map 對象如何解決許多普通對象的問題,包括將鍵轉(zhuǎn)換為字符串。

1. map 可接受任意類型的鍵

如上所述,如果對象的鍵不是字符串或符號,則 JavaScript 會將其隱式轉(zhuǎn)換為字符串。

幸運的是,map 在鍵類型上不存在問題:

const numbersMap = new Map();

numbersMap.set(1, 'one');
numbersMap.set(2, 'two');

[...numbersMap.keys()]; // => [1, 2]

12numbersMap 中的鍵。這些鍵的類型 number 保持不變。

你可以在 map 中使用任何鍵類型:數(shù)字,布爾以及經(jīng)典的字符串和符號。

const booleansMap = new Map();

booleansMap.set(true, "Yep");
booleansMap.set(false, "Nope");

[...booleansMap.keys()]; // => [true, false]

booleansMap 用布爾值作為鍵沒有問題。

同樣,布爾鍵在普通對象中不起作用。

讓我們超越界限:你能把整個對象用作 map 中的鍵嗎?當(dāng)然可以!

1.1 把對象做為鍵

假設(shè)你需要存儲一些與對象相關(guān)的數(shù)據(jù),但是不把這些數(shù)據(jù)附加到對象本身。

不能用普通對象這樣做。

一種解決方法是用一組對象值元組:

const foo = { name: 'foo' };
const bar = { name: 'bar' };

const kindOfMap = [
  [foo, 'Foo related data'],
  [bar, 'Bar related data'],
];

kindOfMap 是一個包含一對對象和關(guān)聯(lián)值的數(shù)組。

這種方法的最大問題是通過鍵訪問值的時間復(fù)雜度為 O(n)  。你必須遍歷整個數(shù)組才能通過鍵獲得所需的值:

function getByKey(kindOfMap, key) {
  for (const [k, v] of kindOfMap) {
    if (key === k) {
      return v;
    }
  }
  return undefined;
}

getByKey(kindOfMap, foo); // => 'Foo related data'

用 WeakMap(Map 的專用版本)你無需為此煩惱。它接受把對象作為鍵。

MapWeakMap 之間的主要區(qū)別是后者允許對作為鍵的對象進行垃圾回收,從而防止內(nèi)存泄漏。

把上面的代碼重構(gòu)為使用 WeakMap 的代碼付出的代價微不足道:

const foo = { name: 'foo' };
const bar = { name: 'bar' };

const mapOfObjects = new WeakMap();

mapOfObjects.set(foo, 'Foo related data');
mapOfObjects.set(bar, 'Bar related data');

mapOfObjects.get(foo); // => 'Foo related data'

Map 相對,WeakMap 僅接受把對象作為鍵,并具有精簡的方法集。

2. map 對鍵名沒有限制

JavaScript 中的任何對象都從其原型對象繼承屬性。普通的 JavaScript 對象也是如此。

如果覆蓋從原型繼承的屬性,則可能會破壞依賴于這些原型屬性的代碼:

function isPlainObject(value) {
  return value.toString() === '[object Object]';
}

const actor = {
  name: 'Harrison Ford',
  toString: 'Actor: Harrison Ford'
};

// Does not work!
isPlainObject(actor); // TypeError: value.toString is not a function

在對象 actor 上定義的屬性 toString 覆蓋了從原型繼承的 toString() 方法。因為它依賴于 toString() 方法,所以這破壞了 isObject()。

檢查普通對象從原型繼承的屬性和方法列表。要避免使用這些名稱定義自定義屬性。

例如,假設(shè)有一個管理某些自定義字段的用戶界面。用戶可以通過指定名稱和值來添加字段:

將自定義字段的狀態(tài)存儲到一個普通對象中會很方便:

const userCustomFields = {
  'color':    'blue',
  'size':     'medium',
  'toString': 'A blue box'
};

但是用戶可能會選擇一個自定義字段名稱,例如 toString(如例中所示), constructor 等,這可能會破壞你的對象。

不要通過接受用戶的輸入在普通對象上創(chuàng)建鍵!

map 則沒有這個問題。鍵的名稱不受限制:

function isMap(value) {
  return value.toString() === '[object Map]';
}

const actorMap = new Map();

actorMap.set('name', 'Harrison Ford');
actorMap.set('toString', 'Actor: Harrison Ford');

// Works!
isMap(actorMap); // => true

不管 actorMap 是否具有名為 toString 的屬性,方法 toString() 都能正常工作。

3. map 是可迭代的

為了遍歷普通對象的屬性,你必須用其他輔助靜態(tài)函數(shù),例如 Object.keys()Object.entries() (在 ES2017 中可用):

const colorsHex = {
  'white': '#FFFFFF',
  'black': '#000000'
};

for (const [color, hex] of Object.entries(colorsHex)) {
  console.log(color, hex);
}
// 'white' '#FFFFFF'
// 'black' '#000000'

Object.entries(colorsHex) 返回從對象提取的鍵值對數(shù)組。

但是,map 本身是可迭代的:

const colorsHexMap = new Map();

colorsHexMap.set('white', '#FFFFFF');
colorsHexMap.set('black', '#000000');

for (const [color, hex] of colorsHexMap) {
  console.log(color, hex);
}
// 'white' '#FFFFFF'
// 'black' '#000000'

colorsHexMap 是可迭代的。你可以在任何可迭代的地方使用它:for() 循環(huán),展開運算符 [...map] 等。

map 還提供了返回迭代的其他方法:map.keys() 遍歷鍵,map.values() 遍歷值。

4. map的大小

普通對象的另一個問題是你無法輕松確定其擁有的屬性數(shù)量:

const exams = {
  'John Smith': '10 points',
  'Jane Doe': '8 points',
};

Object.keys(exams).length; // => 2

要確定 exams 的大小,你必須通過它所有鍵來確定它們的數(shù)量。

map 提供了一種替代方法,通過它的訪問器屬性 size 計算鍵值對:

const examsMap = new Map([
  ['John Smith', '10 points'],
  ['Jane Doe', '8 points'],
]);
  
examsMap.size; // => 2

確定 map 的大小更加簡單:examsMap.size。

5.結(jié)論

普通的 JavaScript 對象通??梢院芎玫乇4娼Y(jié)構(gòu)化數(shù)據(jù)。但是它們有一些限制:

  1. 只能用字符串或符號用作鍵
  2. 自己的對象屬性可能會與從原型繼承的屬性鍵沖突(例如,toString,constructor 等)。
  3. 對象不能用作鍵

所有這些問題都可以通過 map 輕松解決。而且它們提供了諸如迭代器和易于進行大小查找之類的好處。

不要將 map 視為普通對象的替代品,而應(yīng)視為補充。

感謝你能夠認(rèn)真閱讀完這篇文章,希望小編分享如何使用map代替純JS對象內(nèi)容對大家有幫助,同時也希望大家多多支持億速云,關(guān)注億速云行業(yè)資訊頻道,遇到問題就找億速云,詳細(xì)的解決方法等著你來學(xué)習(xí)!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI