返回可变长度通用数组元组的函数
Function returning a variable length generic array tuple
考虑以下几点:
function makeOutputArrayBasedOnInputArray<T>(arrayOfStuff: T[]) {
return arrayOfStuff.map(item => ({ item }));
}
const x = makeOutputArrayBasedOnInputArray([1, 'two']);
const y = x[0] // { item: string | number; }
const z = x[1] // { item: string | number; }
我喜欢的是显示类型:
const y = x[0] // { item: number; }
const z = x[1] // { item: string; }
想象一下 doIt
函数返回一个完全通用且长度可变的元组类型;由函数的输入驱动。所以这也行得通:
const x = makeOutputArrayBasedOnInputArray([7, 'nine', Date]);
const a = x[0] // { item: number; }
const b = x[1] // { item: string; }
const c = x[2] // { item: Date; }
这可能吗?还是我达到了 infer
和元组的限制?
另请参阅: and
根据@A_blop的回答更新:
您好,感谢您的回答!虽然它确实回答了问题,但它依赖于调用者使用 as const
。没有 as const
,唉,它不起作用。
这可能是关键。一些上下文:问题背后的问题是我提出的一个 PR,以改进 react-query 的类型定义:https://github.com/tannerlinsley/react-query/pull/1527/files
目前的变化是这样的:
export function useQueries<
TQueryFnData = unknown,
TError = unknown,
TData = TQueryFnData
>(
queries: UseQueryOptions<TQueryFnData, TError, TData>[]
): UseQueryResult<TData, TError>[] {
// ...
}
这很好,但存在非特定类型推断问题。理想情况下,我们希望这种行为:
const result = useQueries([
{ queryKey: key1, queryFn: () => 1 },
{ queryKey: key2, queryFn: () => 'two' },
])
// result[0].data => number | undefined
// result[1].data => string | undefined
但不需要调用者指定 as const
- 这可能是不可能的?
您可以添加由映射元组类型组成的显式函数 return 类型:
function fn<T extends readonly any[]>(arr: T): {[K in keyof T]: {item: T[K]} } {
return arr.map(item => ({ item })) as any;
}
const x = fn([1, 'two'] as const);
// x: readonly [{ item: 1; }, { item: "two"; }]
const y = x[0] // y: { item: 1; }
const z = x[1] // z: { item: "two"; }
更新:
要从被调用方缩小类型,您可以使用可变元组类型(TS 4.0):
function fn<T extends readonly any[]>(arr: [...T]): // <-- note [...T]
{[K in keyof T]: {item: T[K]} } {...}
考虑以下几点:
function makeOutputArrayBasedOnInputArray<T>(arrayOfStuff: T[]) {
return arrayOfStuff.map(item => ({ item }));
}
const x = makeOutputArrayBasedOnInputArray([1, 'two']);
const y = x[0] // { item: string | number; }
const z = x[1] // { item: string | number; }
我喜欢的是显示类型:
const y = x[0] // { item: number; }
const z = x[1] // { item: string; }
想象一下 doIt
函数返回一个完全通用且长度可变的元组类型;由函数的输入驱动。所以这也行得通:
const x = makeOutputArrayBasedOnInputArray([7, 'nine', Date]);
const a = x[0] // { item: number; }
const b = x[1] // { item: string; }
const c = x[2] // { item: Date; }
这可能吗?还是我达到了 infer
和元组的限制?
另请参阅:
根据@A_blop的回答更新:
您好,感谢您的回答!虽然它确实回答了问题,但它依赖于调用者使用 as const
。没有 as const
,唉,它不起作用。
这可能是关键。一些上下文:问题背后的问题是我提出的一个 PR,以改进 react-query 的类型定义:https://github.com/tannerlinsley/react-query/pull/1527/files
目前的变化是这样的:
export function useQueries<
TQueryFnData = unknown,
TError = unknown,
TData = TQueryFnData
>(
queries: UseQueryOptions<TQueryFnData, TError, TData>[]
): UseQueryResult<TData, TError>[] {
// ...
}
这很好,但存在非特定类型推断问题。理想情况下,我们希望这种行为:
const result = useQueries([
{ queryKey: key1, queryFn: () => 1 },
{ queryKey: key2, queryFn: () => 'two' },
])
// result[0].data => number | undefined
// result[1].data => string | undefined
但不需要调用者指定 as const
- 这可能是不可能的?
您可以添加由映射元组类型组成的显式函数 return 类型:
function fn<T extends readonly any[]>(arr: T): {[K in keyof T]: {item: T[K]} } {
return arr.map(item => ({ item })) as any;
}
const x = fn([1, 'two'] as const);
// x: readonly [{ item: 1; }, { item: "two"; }]
const y = x[0] // y: { item: 1; }
const z = x[1] // z: { item: "two"; }
更新: 要从被调用方缩小类型,您可以使用可变元组类型(TS 4.0):
function fn<T extends readonly any[]>(arr: [...T]): // <-- note [...T]
{[K in keyof T]: {item: T[K]} } {...}