TypeScript:将“in”操作符定义为函数

b4lqfgs4  于 2023-05-01  发布在  TypeScript
关注(0)|答案(2)|浏览(145)

我想创建一个行为与in操作符完全相同的函数,其中使用用户定义的类型保护来缩小类型。
(For一个例子,参见Lodash的has函数。)
对于x表达式中的n,其中n是字符串文字或字符串文字类型,x是联合类型,“true”分支缩小到具有可选或必需属性n的类型,“false”分支缩小到具有可选或缺少属性n的类型。
https://www.typescriptlang.org/docs/handbook/advanced-types.html#using-the-in-operator
我写了一些测试,展示了in操作符的行为以及has函数的预期行为。如何定义一个函数(has),使其行为与这些测试中的in操作符完全相同?

  1. declare const any: any;
  2. type Record = { foo: string; fooOptional?: string };
  3. type Union = { foo: string; fooOptional?: string } | { bar: number; barOptional?: string };
  4. {
  5. const record: Record = any;
  6. if ('foo' in record) {
  7. record; // $ExpectType Record
  8. } else {
  9. record; // $ExpectType never
  10. }
  11. if (has(record, 'foo')) {
  12. record; // $ExpectType Record
  13. } else {
  14. record; // $ExpectType never
  15. }
  16. }
  17. {
  18. const union: Union = any;
  19. if ('foo' in union) {
  20. union; // $ExpectType { foo: string; fooOptional?: string | undefined; }
  21. } else {
  22. union; // $ExpectType { bar: number; barOptional?: string | undefined; }
  23. }
  24. if (has(union, 'foo')) {
  25. union; // $ExpectType { foo: string; fooOptional?: string | undefined; }
  26. } else {
  27. union; // $ExpectType { bar: number; barOptional?: string | undefined; }
  28. }
  29. }
  30. {
  31. const unionWithOptional: { foo: string } | { bar?: number } = any;
  32. if ('bar' in unionWithOptional) {
  33. unionWithOptional; // $ExpectType { bar?: number | undefined; }
  34. } else {
  35. unionWithOptional; // $ExpectType { foo: string; } | { bar?: number | undefined; }
  36. }
  37. if (has(unionWithOptional, 'bar')) {
  38. unionWithOptional; // $ExpectType { bar?: number | undefined; }
  39. } else {
  40. unionWithOptional; // $ExpectType { foo: string; } | { bar?: number | undefined; }
  41. }
  42. }

我最接近解决这个问题的方法是:

  1. type Discriminate<U, K extends PropertyKey> = U extends any
  2. ? K extends keyof U
  3. ? U
  4. : U & Record<K, unknown>
  5. : never;
  6. export const has = <T extends object, K extends PropertyKey>(
  7. source: T,
  8. property: K,
  9. ): source is Discriminate<T, K> =>
  10. property in source;

然而,最后的测试没有通过。
对于上下文,我想这样做的原因是,我的自定义has函数可以在编译时验证键(这是in操作符没有做的事情)。我可以理解这一部分我正在努力的部分只是模仿in执行的收缩。

izj3ouym

izj3ouym1#

如果没有单边或细粒度类型保护(如microsoft/TypeScript#14048中所要求的),支持保护的true和false端的独立行为,目前是不可能的。
你所要求的是接受一个联合类型{ foo: string } | { bar?: number },并提出一个用户定义的类型保护,其中真分支将其缩小为{ bar?: number },假分支将其保留为完整的联合{ foo: string } | { bar?: number }
阅读checker.ts的第19788到19809行中的getNarrowedType()函数,编译器的类型检查器,我认为这不可能发生。
现在,将用户定义的(x: any) => x is G类型的类型保护应用于 union type T的值,编译器当前将对类型保护的true分支和false分支的union类型进行 * 分区。所以在true分支中你会得到类似于Extract<T, G>的东西,在false分支中你会得到类似于Exclude<T, G>的东西。如果一个联合成员存在于真分支中,它将不存在于假分支中,反之亦然。
只有当Tnot 联合类型时,真分支和假分支窄化才可能不互相排斥(有时你会得到真分支的T & G,而假分支只有T)。
这并不完美(按照ms/TS#31156),但它就是这样。

iyr7buue

iyr7buue2#

has函数可以用一种缩小其参数类型的方式来编写,至少在键是可选的并且可以缩小为非可选的情况下:

  1. function has<T,K extends keyof T>(obj: T, key: K):
  2. obj is T & Record<K, Exclude<T[K], undefined>>
  3. {
  4. return obj[key] !== undefined;
  5. }
  6. type Example = { a: number, b?: number };
  7. let obj: Example = { a: 1, b: 2 };
  8. // inferred obj.b : number|undefined
  9. if(has(obj, 'b')) {
  10. // inferred obj.b : number
  11. }

但是,您对Union类型的要求并不合理。请考虑以下示例:

  1. type UnsoundUnion = { a: number, b: number } | { c: number, d: number };
  2. let u: UnsoundUnion = { a: 1, c: 2, d: 3 };
  3. if(has(u, 'a')) {
  4. // it would be wrong to infer u.b : number
  5. }
展开查看全部

相关问题