typescript 为什么封装到泛型省略行为与非封装不同?

llmtgqce  于 2023-03-13  发布在  TypeScript
关注(0)|答案(1)|浏览(157)

当我尝试忽略泛型外的属性时,它工作正常,但在泛型内会中断。基本上,问题是Omit<Thing, 'key1' | 'key2'>变成了Omit<Thing, 'key1'> | Omit<Thing, 'key2'>,并且typescript不允许使用Omit<Thing, 'key1'>Omit<Thing, 'key2'>中的偶数键
现场演示: typescript Playground

  1. export class AccessScope {
  2. id!: number;
  3. type!: string;
  4. createdAt!: Date;
  5. updatedAt!: Date;
  6. }
  7. type KeysWhichShouldBeOmitted = "id" | "createdAt" | "updatedAt";
  8. type AccessScopeWithOmittedId = Pick<AccessScope, "type" | "createdAt" | "updatedAt">;
  9. type OmitIfNotNull<
  10. Entity,
  11. Holder,
  12. KeysToOmit,
  13. > = KeysToOmit extends keyof Entity
  14. ? Omit<Holder, KeysToOmit>
  15. : Holder;
  16. type brokenType1 = OmitIfNotNull<
  17. AccessScope,
  18. AccessScopeWithOmittedId,
  19. KeysWhichShouldBeOmitted
  20. >
  21. // Type '{ type: string; }' is not assignable to type 'brokenType1'.
  22. // Property 'createdAt' is missing in type '{ type: string; }' but required in type 'Omit<Pick<AccessScope, "createdAt" | "updatedAt" | "type">, "updatedAt">'.(2322)
  23. const brokenVar1: brokenType1 = {
  24. type: 'asd'
  25. }
  26. // brokenType2 is fully visible version of brokenType1
  27. type brokenType2 = Omit<Pick<AccessScope, "type" | "createdAt" | "updatedAt"> & Record<never, never>, "id">
  28. | Omit<Pick<AccessScope, "type" | "createdAt" | "updatedAt"> & Record<never, never>, "createdAt">
  29. | Omit<Pick<AccessScope, "type" | "createdAt" | "updatedAt"> & Record<never, never>, "updatedAt">
  30. // Type '{ type: string; }' is not assignable to type 'brokenType1'.
  31. // Property 'createdAt' is missing in type '{ type: string; }' but required in type 'Omit<Pick<AccessScope, "createdAt" | "updatedAt" | "type">, "updatedAt">'.(2322)
  32. const brokenVar2: brokenType2 = {
  33. type: 'asd'
  34. }
  35. type normalType = KeysWhichShouldBeOmitted extends keyof AccessScope
  36. ? Omit<AccessScopeWithOmittedId, KeysWhichShouldBeOmitted>
  37. : AccessScopeWithOmittedId
  38. const normalType: normalType = {
  39. type: 'asd'
  40. }

我尝试添加了扩展OmitIfNotNull参数的各种奇怪变体

  1. type OmitIfNotNull<
  2. Entity extends Holder,
  3. Holder extends Record<string, any>,
  4. KeysToOmit extends keyof Entity,
  5. > = KeysToOmit extends keyof Entity
  6. ? Omit<Holder, KeysToOmit>
  7. : Holder;
gblwokeq

gblwokeq1#

此行为是由条件类型的分布特性引起的。您可以通过将extends子句的两侧括在方括号中来禁用此行为:

  1. export class AccessScope {
  2. id!: number;
  3. type!: string;
  4. createdAt!: Date;
  5. updatedAt!: Date;
  6. }
  7. type KeysWhichShouldBeOmitted = "id" | "createdAt" | "updatedAt";
  8. type AccessScopeWithOmittedId = Pick<AccessScope,
  9. "type" | "createdAt" | "updatedAt">;
  10. type OmitIfNotNull<
  11. Entity,
  12. Holder,
  13. KeysToOmit,
  14. > = [KeysToOmit] extends [keyof Entity]
  15. ? Omit<Holder, KeysToOmit>
  16. : Holder;
  17. type noLongerBrokenType = OmitIfNotNull<
  18. AccessScope,
  19. AccessScopeWithOmittedId,
  20. KeysWhichShouldBeOmitted
  21. >
  22. const noLongerBrokenVar: noLongerBrokenType = {
  23. type: 'asd'
  24. }

Playground链接

展开查看全部

相关问题