2022年7月18日

Object.keys,values,entries

对各个数据结构的学习至此告一段落,下面让我们讨论一下如何迭代它们。

在前面的章节中,我们认识了 map.keys()map.values()map.entries() 方法。

这些方法是通用的,有一个共同的约定来将它们用于各种数据结构。如果我们创建一个我们自己的数据结构,我们也应该实现这些方法。

它们支持:

  • Map
  • Set
  • Array

普通对象也支持类似的方法,但是语法上有一些不同。

Object.keys,values,entries

对于普通对象,下列这些方法是可用的:

……但是请注意区别(比如说跟 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),它会返回 所有 键。

转换对象

对象缺少数组存在的许多方法,例如 mapfilter 等。

如果我们想应用它们,那么我们可以使用 Object.entries,然后使用 Object.fromEntries

  1. 使用 Object.entries(obj)obj 获取由键/值对组成的数组。
  2. 对该数组使用数组方法,例如 map,对这些键/值对进行转换。
  3. 对结果数组使用 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.valuesfor..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.valuesreduce 来求和:

// 使用 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;
}

使用沙箱的测试功能打开解决方案。

教程路线图

评论

在评论之前先阅读本内容…
  • 如果你发现教程有错误,或者有其他需要修改和提升的地方 — 请 提交一个 GitHub issue 或 pull request,而不是在这评论。
  • 如果你对教程的内容有不理解的地方 — 请详细说明。
  • 使用 <code> 标签插入只有几个词的代码,插入多行代码可以使用 <pre> 标签,对于超过 10 行的代码,建议你使用沙箱(plnkrJSBincodepen…)