TypeScript中遍历对象键的方法
创始人
2025-07-07 15:00:56
0

前言

在日常的TypeScript开发中,经常需要遍历对象的键来执行各种操作。然而,使用Object.keys时可能会遇到一些类型相关的困扰,因为它返回的是一个字符串数组,而不是期望的键的联合类型。这可能导致在代码中引入一些不安全的类型转换。在本文中,我们将深入研究这个问题,并提供几种解决方案,以便在遍历对象键时更安全、更灵活地操作。

背景

使用Object.keys进行遍历并不能按照预期工作。这是因为Object.keys返回一个字符串数组,而不是包含所有键的联合类型。这是设计上的考虑,不会改变。

function printUser(user: User) {
  Object.keys(user).forEach((key) => {
    // 不起作用!
    console.log(user[key]);
    // 报错:属性“key”在类型“User”上不存在。
  });
}

在适当的位置进行keyof typeof类型转换可以解决这个问题:

const user = {
  name: "Daniel",
  age: 26,
};


const keys = Object.keys(user);


keys.forEach((key) => {
  // 不再报错!
  console.log(user[key as keyof typeof user]);
});

通过自定义类型断言,可以在行内缩小类型:

function isKey(
  x: T,
  k: PropertyKey
): k is keyof T {
  return k in x;
}


keys.forEach((key) => {
  if (isKey(user, key)) {
    console.log(user[key]);
    // key现在被缩小为 "name" | "age"
  }
});

Object.keys

问题在于使用Object.keys似乎无法按照期望的方式工作。这是因为它不会返回你需要的类型。

const user = {
  name: "Daniel",
  age: 26,
};

const keys = Object.keys(user);

// keys的类型是 string[]

这意味着你不能使用键来访问对象上的值:

const nameKey = keys[0];

user[nameKey];
// 报错:属性“nameKey”在类型“{ name: string; age: number; }”上不存在。

TypeScript之所以返回字符串数组,是因为它的对象类型是开放的。在许多情况下,TS无法保证由Object.keys返回的键实际上存在于对象上 - 因此将它们扩展为字符串是唯一合理的解决方案。

for...in

如果尝试使用for...in循环,同样会失败,原因是键被推断为字符串,就像Object.keys一样。

function printUser(user: User) {
  for (const key in user) {
    console.log(user[key]);
    // 报错:属性“key”在类型“User”上不存在。
  }
}

但在许多情况下,你可能确信自己完全了解对象的形状。

那么,怎么办呢?

解决方案1:转换为keyof typeof

第一种选择是使用keyof typeof将键转换为更具体的类型。

const user = {
  name: "Daniel",
  age: 26,
};


const keys = Object.keys(user) as Array;


keys.forEach((key) => {
  // 不再报错!
  console.log(user[key]);
});

在索引对象时也可以进行转换。

const keys = Object.keys(user);


keys.forEach((key) => {
  console.log(user[key as keyof typeof user]);
});

然而,as在任何形式中通常是不安全的,这也不例外。

const user = {
  name: "Daniel",
  age: 26,
};


const nonExistentKey = "id" as keyof typeof user;


// 没有错误!
const value = user[nonExistentKey];
// as是一个强大的工具,它允许我们在类型上欺骗TypeScript

解决方案2:类型断言

通过使用isKey助手,可以在索引之前检查键是否实际存在于对象中。

function isKey(
  x: T,
  k: PropertyKey
): k is keyof T {
  return k in x;
}


keys.forEach((key) => {
  if (isKey(user, key)) {
    console.log(user[key]);
    // key现在被缩小为 "name" | "age"
  }
});

解决方案3:泛型函数

再来看一个略微奇怪的解决方案。在泛型函数内部,使用in运算符将类型缩小到键。

function printEachKey(obj: T) {
  for (const key in obj) {
    console.log(obj[key]);
    // key的类型被缩小为Extract
  }
}


// 每个键都被打印出来!
printEachKey({
  name: "Daniel",
  age: 26,
});

解决方案4:将Object.keys包装在函数中

另一种解决方案是将Object.keys包装在一个返回转换类型的函数中。

const objectKeys = (obj: T) => {
  return Object.keys(obj) as Array;
};


const keys = objectKeys({
  name: "Daniel",
  age: 26,
});


console.log(keys);
// keys的类型是("name" | "age")[]

这可能是最容易被滥用的解决方案 - 将转换隐藏在函数中使其更有吸引力,可能导致人们在不考虑的情况下使用它。

结论

本文介绍了一些解决方案,从简单的类型转换到更智能的类型谓词,帮助我们更安全、更可靠地进行对象键的遍历。选择哪种方法取决于项目的需求和个人偏好,但总体而言,通过了解这些技术,我们可以更好地利用TypeScript的类型系统,提高代码的可维护性和安全性。


相关内容

热门资讯

PHP新手之PHP入门 PHP是一种易于学习和使用的服务器端脚本语言。只需要很少的编程知识你就能使用PHP建立一个真正交互的...
网络中立的未来 网络中立性是什... 《牛津词典》中对“网络中立”的解释是“电信运营商应秉持的一种原则,即不考虑来源地提供所有内容和应用的...
各种千兆交换机的数据接口类型详... 千兆交换机有很多值得学习的地方,这里我们主要介绍各种千兆交换机的数据接口类型,作为局域网的主要连接设...
全面诠释网络负载均衡 负载均衡的出现大大缓解了服务器的压力,更是有效的利用了资源,提高了效率。那么我们现在来说一下网络负载...
什么是大数据安全 什么是大数据... 在《为什么需要大数据安全分析》一文中,我们已经阐述了一个重要观点,即:安全要素信息呈现出大数据的特征...
如何允许远程连接到MySQL数... [[277004]]【51CTO.com快译】默认情况下,MySQL服务器仅侦听来自localhos...
如何利用交换机和端口设置来管理... 在网络管理中,总是有些人让管理员头疼。下面我们就将介绍一下一个网管员利用交换机以及端口设置等来进行D...
P2P的自白|我不生产内容,我... 现在一提起P2P,人们就会联想到正在被有关部门“围剿”的互联网理财服务。×租宝事件使得劳...
Intel将Moblin社区控... 本周二,非营利机构Linux基金会宣布,他们将担负起Moblin社区的管理工作,而这之前,Mobli...
施耐德电气数据中心整体解决方案... 近日,全球能效管理专家施耐德电气正式启动大型体验活动“能效中国行——2012卡车巡展”,作为该活动的...