2018年2月20日更新:将其发布为issue on GitHub。
2月28日更新:关闭该问题,转而支持a new one。
在下面的代码片段中,如果查看最后两行,TypeScript会在第一行中显示错误,并在第二行中正确推断类型,尽管差异只是函数通过管道传递的顺序。
const pipe = <A, B, C>(
x: A,
a: (x: A) => B,
b: (x: B) => C,
) => b(a(x));
// This just calls the function passed as argument.
const call = <A, B>(f: (x: A) => B) => (x: A) => f(x)
const a = pipe(1, x => x + 1, call(x => x + 1));
const b = pipe(1, call(x => x + 1), x => x + 1);
我在严格模式下使用TypeScript 2.7.1(包括strictFunctionTypes),但严格模式似乎并不重要。 Here is this snippet on TypeScript playground
这是我在使用RxJS时经常遇到的一个问题,因为在RxJS中有一个类似的pipe
方法,我在使用创建运算符(如obs => merge(obs, otherObs)
)时将箭头函数传递给它。通常通过指定参数类型很容易解决这个问题,但我想了解这背后的逻辑。为什么TypeScript能够在一种情况下推断出类型但在另一种情况下却不能推断?
答案 0 :(得分:2)
这个问题仍然很突出,但是现在我想发布一个我想出的解决方法,因为我发现它在我的经验中非常有用。
创建一个这样的实用函数:
export const call = <A, B>(f: (x: A) => B) => f;
(与问题中的函数相同,一个将一元函数作为参数的标识函数),然后每当您在管道中遇到箭头函数问题时,请尝试使用此实用程序函数将其括起来。与直觉相反,TS无法解决这个问题:
const a = pipe(1, x => x + 1, call(x => x + 1));
但它可以解决这个问题:
const c = pipe(1, call(x => x + 1), call(x => x + 1));
(不知道为什么)。稍后当问题得到解决时(继续进行投票:https://github.com/Microsoft/TypeScript/issues/22081),您将能够轻松找到对call
的所有引用并将其删除。