gpt4 book ai didi

node.js - TypeScript 类型在基于其他参数的回调中缩小函数参数

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

我正在处理经典 Node 回调。示例:

myFunction('foo', (err: Error|null, data?: Buffer) =>{
if (err) {
// typeof err is Error
// typeof data is Buffer|undefined
} else {
// typeof err is null;
// typeof data is Buffer|undefined;
}
});

我正在尝试定义我自己的回调接受函数 myFunction。我正在努力实现两件事:

  1. 我希望能够根据 err 的类型推断出 data 的类型>
  2. 我希望能够隐式推断输入参数的类型

例子:

// types of err and data are inferred from readFile
readFile('foo', (err, data) => {
if (err) {
// typeof err is Error
// typeof data is undefined
} else {
// typeof err is null
// typeof data is Buffer
}
}

有什么方法可以在当前的 typescript 中实现其中任何一个吗?

最佳答案

按照惯例,您会使用 discriminated union 的单个参数type 而不是两个联合类型参数,如 errdata,它们的类型相互关联。

TypeScript 对相关表达式 的支持确实不多;见microsoft/TypeScript#30581 .也就是说,没有很好的方法来告诉编译器虽然 errError | 类型。 nulldataBuffer | undefinederrdata 的某些类型组合是不可能的。您可以使用 control flow analysis检查 err 的类型,但它不会影响 data 的感知类型...编译器错误地假定它们是独立的表达式。

这是我能得到的最接近的;它大量使用 tuples in rest and spread expressions你真的必须与编译器斗争才能让它发生:

declare function myFunction(
someString: string,
someCallback: (...args: [Error, undefined] | [null, Buffer]) => void
): void;

someCallback 的类型恰好是两个参数的函数,这两个参数要么是类型[Error, undefined],要么是类型化的[null, Buffer] 。现在您可以使用双参数回调调用它,但是您会遇到与您已经遇到的完全相同的问题:检查 err 不会对 做任何事情数据:

// can't call it this way
myFunction("oops", (err, data) => {
err; // Error | null
data; // Buffer | undefined
if (err) {
data; // still Buffer | undefined 😟
}
});

相反,您还必须在回调实现中使用剩余参数:

myFunction("foo", (...errData) => {
if (errData[0]) {
const [err, data] = errData;
err; // Error
data; // undefined
} else {
const [err, data] = errData;
err; // null
data; // Buffer
}
});

这是可行的,因为当您检查 errData[0] 时,您正在检查元组对象的单个属性,并且编译器将使用控制流分析将 errData 缩小到两种已知类型之一。您只能将 errData 分解为 errdata after 检查。


我强烈建议您考虑在回调中切换到单个可区分的联合参数。看看这个解决方案是多么简单:

type Param = { type: "Error"; err: Error } | { type: "Data"; data: Buffer };

declare function myFunction(
someString: string,
someCallback: (param: Param) => void
): void;

myFunction("foo", param => {
if (param.type === "Error") {
param.err; // Error
} else {
param.data; // Buffer
}
});

这正是您想要的,无需战斗。


Link to code

关于node.js - TypeScript 类型在基于其他参数的回调中缩小函数参数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57047629/

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