gpt4 book ai didi

typescript - 为什么这个 Typescript 泛型类型推断会失败?

转载 作者:搜寻专家 更新时间:2023-10-30 21:32:21 26 4
gpt4 key购买 nike

我有 2 个版本的数据模型(称它们为 AB),我希望能够在这些实例上执行函数,无论它们是 A 还是 B。函数本身特定于类型 A 和 B。

我想出了一些方法来解决这个问题,但它们都涉及声明一个泛型类型(即 Functions<T> ),而我目前的设置无法轻松做到这一点。

type A = {
a: string;
}

type B = {
b: string;
}

type AFunctions = {
make: () => A;
do: (obj: A) => void;
}

type BFunctions = {
make: () => B;
do: (obj: B) => void;
}

type Bundle<
Fns extends AFunctions | BFunctions,
T extends ReturnType<Fns['make']> = ReturnType<Fns['make']>
> = {
obj: T,
fns: Fns,
}

function doIt<M extends AFunctions | BFunctions>(bundle: Bundle<M>) {
bundle.fns.do(bundle.obj);
}

Typescript playground

在线bundle.fns.do(bundle.obj); ,我收到 typescript 错误:Argument of type 'ReturnType<M["make"]>' is not assignable to parameter of type 'A & B'

我希望 doIt是类型安全的,因为 bundle.objbundle.fns.do 上的参数类型相同.这里出了什么问题?有没有办法在不引入通用 Functions<T> 的情况下解决这个问题? ?


我也可以通过添加 { type: 'a' } 来解决它和 { type: 'b' }相应的参数 Bundle s,然后检查:

if (bundle.type === 'a') {
bundle.fns.do(bundle.obj);
} else if (bundle.type === 'b') {
bundle.fns.do(bundle.obj);
}

但这种冗余并不理想。

我认为这与缩小泛型类型的问题有关:https://github.com/microsoft/TypeScript/issues/17859

最佳答案

我认为编译器的提示是对的。请考虑以下事项:

let
func : AFunctions | BFunctions = {
'make' : function() : A { return {'a': "A"} },
'do' : function(_ : A) { }
},
someB : B = { 'b' : "B" },
bundle : Bundle<AFunctions | BFunctions> = {
'obj' : someB,
'fns' : func,
}

这个类型检查,但是 'do' 显然不能用参数 'obj' 调用。根本问题在于,在 bundle 中,类型 T 被推断为 A | B,而不是 AB,这取决于 make 的类型,正如我认为您所期望的那样。

您想要实现的目标并不是很清楚。特别是,不清楚为什么不能声明泛型类型,因为这似乎正是您所需要的:

type GenericBundle<X> = { obj : X, do : (obj : X) => void };
type AFunctions = GenericBundle<A>;
type BFunctions = GenericBundle<B>;
type Bundle = AFunctions | BFunctions;

function doIt<X>(bundle: GenericBundle<X>) {
bundle.do(bundle.obj);
}

let
someA : A = { 'a' : "A" },
someB : B = { 'b' : "B" },
bundle1 : Bundle = {
'obj' : someA,
'do' : function(_ : A) { },
},
bundle2 : Bundle = {
'obj' : someB,
'do' : function(_ : B) { },
},
bundle3_wrong : Bundle = { // doesn't typecheck
'obj' : someA,
'do' : function(_ : B) { },
},
bundle4_wrong : Bundle = { // doesn't typecheck
'obj' : someB,
'do' : function(_ : A) { },
};

doIt(bundle1);
doIt(bundle2);

关于typescript - 为什么这个 Typescript 泛型类型推断会失败?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57878261/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com