如何在Typescript中为嵌套对象中的键创建一个守卫?

huangapple go评论99阅读模式
英文:

How to create a guard in Typescript for keys in a nested object?

问题

以下是代码的中文翻译部分:

  1. type PossibleKeys = 'a' | 'b' | 'c';
  2. type MappedObject = {
  3. test: {
  4. [key in PossibleKeys]?: { foo: boolean };
  5. }
  6. }
  7. const mappedObject: MappedObject = {
  8. test: {
  9. a: { foo: true },
  10. b: { foo: false },
  11. }
  12. }
  13. function mappedObjectHasKeyDefined(possibleKey: PossibleKeys): possibleKey is keyof MappedObject['test'] {
  14. return possibleKey in mappedObject.test;
  15. }
  16. const c: PossibleKeys = 'c';
  17. const array: PossibleKeys[] = ['a', 'b', 'c'];
  18. array.forEach((newT: PossibleKeys) => {
  19. // 所有这些类型保护似乎都不起作用
  20. if(
  21. mappedObject.test[newT] !== undefined
  22. && newT in mappedObject.test
  23. && mappedObject.test
  24. && mappedObject.test[newT]
  25. && Object.prototype.hasOwnProperty.call(mappedObject.test, newT)
  26. && mappedObjectHasKeyDefined(newT)
  27. ) {
  28. // 对象可能为 'undefined'。
  29. const a = mappedObject.test[newT].foo;
  30. }
  31. })

链接至 TypeScript Playground

英文:

I have a nested object that contains an object with multiple optional properties. I would like to create a typeguard to prevent the Object is possibly undefined. I do not know if this is possible or that I should cast it.

  1. type PossibleKeys = 'a' | 'b' | 'c';
  2. type MappedObject = {
  3. test: {
  4. [key in PossibleKeys]?: { foo: boolean };
  5. }
  6. }
  7. const mappedObject: MappedObject = {
  8. test: {
  9. a: { foo: true },
  10. b: { foo: false },
  11. }
  12. }
  13. function mappedObjectHasKeyDefined(possibleKey: PossibleKeys): possibleKey is keyof MappedObject['test'] {
  14. return possibleKey in mappedObject.test;
  15. }
  16. const c: PossibleKeys = 'c';
  17. const array: PossibleKeys[] = ['a', 'b', 'c'];
  18. array.forEach((newT: PossibleKeys) => {
  19. // all of these guards somehow do not seem to work
  20. if(
  21. mappedObject.test[newT] !== undefined
  22. && newT in mappedObject.test
  23. && mappedObject.test
  24. && mappedObject.test[newT]
  25. && Object.prototype.hasOwnProperty.call(mappedObject.test, newT)
  26. && mappedObjectHasKeyDefined(newT)
  27. ) {
  28. // Object is possibly 'undefined'.
  29. const a = mappedObject.test[newT].foo;
  30. }
  31. })

Link to the typescript playground

答案1

得分: 0

TypeScript当前不会在obj[key]形式的属性访问的外观类型上进行缩小,当key不是单一字符串字面类型时。如果key只是string类型,或者如果它是字符串字面类型的联合类型(如PossibleKeys),或者如果它是泛型类型,编译器将不会缩小属性的类型。这仍然是一个待解决的功能请求,跟踪在microsoft/TypeScript#10530

目前的解决方法是将属性复制到一个新变量中,因为变量可以通过检查来缩小类型。以下是一种方法:

  1. array.forEach((newT: PossibleKeys) => {
  2. const t = mappedObject.test; // 可行
  3. const u = t[newT]; // 可行
  4. if (u) {
  5. const a = u.foo // 可行
  6. }
  7. })

代码的Playground链接

英文:

TypeScript doesn't currently narrow the apparent types of property accesses of the form obj[key] when key is not of a single string literal type. If key is just of type string, or if it's a union of string literal types (like PossibleKeys) or if it's a generic type, the compiler will not narrow the type of the property. This is still an open feature request, tracked at microsoft/TypeScript#10530.

Currently the workaround is to copy the property into a new variable, since variables can be narrowed via checks. Here's one way to do it:

  1. array.forEach((newT: PossibleKeys) => {
  2. const t = mappedObject.test; // okay
  3. const u = t[newT]; // okay
  4. if (u) {
  5. const a = u.foo // okay
  6. }
  7. })

Playground link to code

huangapple
  • 本文由 发表于 2023年7月4日 23:25:58
  • 转载请务必保留本文链接:https://go.coder-hub.com/76614044.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定