对各个数据结构的学习至此告一段落,下面让我们讨论一下如何迭代它们。
在前面的章节中,我们认识了 map.keys()
,map.values()
和 map.entries()
方法。
这些方法是通用的,有一个共同的约定来将它们用于各种数据结构。如果我们创建一个我们自己的数据结构,我们也应该实现这些方法。
它们支持:
Map
Set
Array
普通对象也支持类似的方法,但是语法上有一些不同。
对于普通对象,下列这些方法是可用的:
Object.keys(obj) —— 返回一个包含该对象所有的键的数组。
Object.values(obj) —— 返回一个包含该对象所有的值的数组。
Object.entries(obj) —— 返回一个包含该对象所有 [key, value] 键值对的数组。
……但是请注意区别(比如说跟 map 的区别):
Map | Object | |
---|---|---|
调用语法 | map.keys() | Object.keys(obj) ,而不是 obj.keys() |
返回值 | 可迭代对象 | “真正的”数组 |
第一个区别是,对于对象我们使用的调用语法是 Object.keys(obj)
,而不是 obj.keys()
。
为什么会这样?主要原因是灵活性。请记住,在 JavaScript 中,对象是所有复杂结构的基础。因此,我们可能有一个自己创建的对象,比如 data
,并实现了它自己的 data.values()
方法。同时,我们依然可以对它调用 Object.values(data)
方法。
第二个区别是 Object.*
方法返回的是“真正的”数组对象,而不只是一个可迭代对象。这主要是历史原因。
举个例子:
let user = { name: "John", age: 30 };
Object.keys(user) = ["name", "age"]
Object.values(user) = ["John", 30]
Object.entries(user) = [ ["name","John"], ["age",30] ]
这里有一个使用 Object.values
来遍历属性值的例子:
let user = { name: "John", age: 30 }; // 遍历所有的值 for (let value of Object.values(user)) { alert(value); // John, then 30 }
Object.keys/values/entries 会忽略 symbol 属性
就像 for..in
循环一样,这些方法会忽略使用 Symbol(...)
作为键的属性。
通常这很方便。但是,如果我们也想要 Symbol 类型的键,那么这儿有一个单独的方法 Object.getOwnPropertySymbols,它会返回一个只包含 Symbol 类型的键的数组。另外,还有一种方法 Reflect.ownKeys(obj),它会返回 所有 键。
对象缺少数组存在的许多方法,例如 map
和 filter
等。
如果我们想应用它们,那么我们可以使用 Object.entries
,然后使用 Object.fromEntries
:
使用 Object.entries(obj)
从 obj
获取由键/值对组成的数组。
对该数组使用数组方法,例如 map
,对这些键/值对进行转换。
对结果数组使用 Object.fromEntries(array)
方法,将结果转回成对象。
例如,我们有一个带有价格的对象,并想将它们加倍:
let prices = { banana: 1, orange: 2, meat: 4, }; let doublePrices = Object.fromEntries( // 将价格转换为数组,将每个键/值对映射为另一对 // 然后通过 fromEntries 再将结果转换为对象 Object.entries(prices).map(entry => [entry[0], entry[1] * 2]) ); alert(doublePrices.meat); // 8
乍一看,可能感觉有点困难,但是使用一两次之后就很容易理解了。我们可以通过这种方式建立强大的转换链。
重要程度: 5
有一个带有任意数量薪水的 salaries
对象。
编写函数 sumSalaries(salaries)
,该函数使用 Object.values
和 for..of
循环返回所有薪水的总和。
如果 salaries
是空对象,那么结果必须是 0
。
举个例子:
let salaries = { "John": 100, "Pete": 300, "Mary": 250 }; alert( sumSalaries(salaries) ); // 650
打开带有测试的沙箱。
function sumSalaries(salaries) { let sum = 0; for (let salary of Object.values(salaries)) { sum += salary; } return sum; // 650 } let salaries = { "John": 100, "Pete": 300, "Mary": 250 }; alert( sumSalaries(salaries) ); // 650
还有另外一种可选方案,即使用 Object.values
和 reduce
来求和:
// 使用 reduce 方法遍历 salaries 数组 // 把它们加起来 // 返回最终结果 function sumSalaries(salaries) { return Object.values(salaries).reduce((a, b) => a + b, 0) // 650 }
使用沙箱的测试功能打开解决方案。
重要程度: 5
写一个函数 count(obj)
,该函数返回对象中的属性的数量:
let user = { name: 'John', age: 30 }; alert( count(user) ); // 2
试着使代码尽可能简短。
P.S. 忽略 Symbol 类型属性,只计算“常规”属性。
打开带有测试的沙箱。
function count(obj) { return Object.keys(obj).length; }
使用沙箱的测试功能打开解决方案。