如何动态引用 TypeScript 接口属性的值?

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

How to dynamically refer to TypeScript interface property's value?

问题

请看这个愚蠢的例子:

  1. interface Building {
  2. rooms: number
  3. kitchen: {
  4. hasOven: true
  5. }
  6. bathroom: {
  7. numberOfSinks: number
  8. }
  9. }
  10. interface Select<T> {
  11. name: keyof T
  12. fields: (keyof T[this.name])[] // <-- 这不是有效的代码。我应该怎么做?
  13. }
  14. const x: Select<Building> = {
  15. name: 'kitchen',
  16. fields: ['hasOven'] // 因为我指定了 'kitchen',所以它只允许 'hasOven'
  17. }
  18. const y: Select<Building> = {
  19. name: 'bathroom',
  20. fields: ['numberOfSinks'] // 因为我指定了 'bathroom',所以它只允许 'numberOfSinks'
  21. }

上面的代码不起作用。

如何使 Select<T> 接口能够动态引用 name 属性的值,并使用它来确定可用的键值?

非常感谢!

英文:

Consider this silly example:

  1. interface Building {
  2. rooms: number
  3. kitchen: {
  4. hasOven: true
  5. }
  6. bathroom: {
  7. numberOfSinks: number
  8. }
  9. }
  10. interface Select<T> {
  11. name: keyof T
  12. fields: (keyof T[this.name])[] // <-- This is not valid code. What should I do instead?
  13. }
  14. const x: Select<Building> = {
  15. name: 'kitchen',
  16. fields: ['hasOven'] // Because I specified 'kitchen', it only allows 'hasOven'
  17. }
  18. const y: Select<Building> = {
  19. name: 'bathroom',
  20. fields: ['numberOfSinks'] // Because I specified 'bathroom', it only allows 'numberOfSinks'
  21. }

The above code does not work.

How do I make the Select<T> interface to be able to refer to the name property's value dynamically and use that to determine the available key values?

Thanks so much in advance!

答案1

得分: 2

以下是代码的翻译部分:

  1. 无法在不引入第二个泛型类型的情况下完成:
  2. interface Select<T, K extends keyof T> {
  3. name: K
  4. fields: (keyof T[K])[]
  5. }
  6. 所以现在
  7. const a: Select<Building, "bathroom"> = {
  8. name: 'bathroom',
  9. fields: ['hasOven'] // 不行!
  10. }
  11. const b: Select<Building, "steering-wheel"> = { // 不行!
  12. name: 'steering-wheel',
  13. fields: ['hasOven'] // 不行!
  14. }
  15. const c: Select<Building, "bathroom"> = {
  16. name: 'steering-wheel', // 不行!
  17. fields: ['numberOfSinks']
  18. }

Playground链接

英文:

Can't be done without introducing a second generic type:

  1. interface Select&lt;T, K extends keyof T&gt; {
  2. name: K
  3. fields: (keyof T[K])[]
  4. }

so now

  1. const a: Select&lt;Building, &quot;bathroom&quot;&gt; = {
  2. name: &#39;bathroom&#39;,
  3. fields: [&#39;hasOven&#39;] // NO!
  4. }
  5. const b: Select&lt;Building, &quot;steering-wheel&quot;&gt; = { // NO!
  6. name: &#39;steering-wheel&#39;,
  7. fields: [&#39;hasOven&#39;] // NO!
  8. }
  9. const c: Select&lt;Building, &quot;bathroom&quot;&gt; = {
  10. name: &#39;steering-wheel&#39;, // NO!
  11. fields: [&#39;numberOfSinks&#39;]
  12. }

Playground Link

huangapple
  • 本文由 发表于 2023年3月10日 02:01:17
  • 转载请务必保留本文链接:https://go.coder-hub.com/75688417.html
匿名

发表评论

匿名网友

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

确定