我正在使用这些接口(interface)
/**
* Interface for entities
*/
export interface Entity extends Object {
readonly id: EntityId;
}
/**
* Interface for collection-based state
*/
export interface Collection<T extends Entity> extends Object {
readonly entities: { [key: string]: T };
readonly ids: EntityId[];
}
我正在编写一个辅助函数,它给定路径,增加集合中实体的相应值。
示例:
Collection.increment(myCollection, 'some-id', ['stats', 'total']);
这些是我当前的输入
export function increment<
C extends Collection<any>,
E = C extends Collection<infer U> ? U : never,
K1 extends keyof E = keyof E,
V1 extends E[K1]= E[K1]
>(collection: C, entityId: string, path: K1 | [K1]): C
export function increment<
C extends Collection<any>,
E = C extends Collection<infer U> ? U : never,
K1 extends keyof E = keyof E,
V1 = Exclude<E[K1], void>,
K2 extends keyof V1 = keyof V1,
V2 extends V1[K2]= V1[K2]
>(collection: C, entityId: string, path: K1 | [K1] | [K1, K2]): C
对于这个例子,类型非常冗长,因为它们用于对传入的内容进行类型检查
上述类型有效,唯一的问题是非数字类型的路径仍然有效。
I tried something like the below to force it to only allow numbers
export function increment<
C extends Collection<any>,
E = C extends Collection<infer U> ? U : never,
K1 extends keyof E = keyof E,
V1 extends E[K1]= E[K1]
>(collection: C, entityId: string, path: K1 | [K1]): C
export function increment<
C extends Collection<any>,
E = C extends Collection<infer U> ? U : never,
K1 extends keyof E = keyof E,
V1 = Extract<Exclude<E[K1], void>, number>, // Note the Extract
K2 extends keyof V1 = keyof V1,
V2 extends V1[K2]= V1[K2]
>(collection: C, entityId: string, path: K1 | [K1] | [K1, K2]): C
但是上面的内容并没有按预期强制执行约束
有什么想法吗?
最佳答案
您需要按属性类型过滤可用键。从 Omit
中获取线索type 我们可以使用条件类型来创建一个类型,该类型将根据属性类型过滤掉属性。
type FilterKeysByType<T, U> = ({[P in keyof T]: T[P] extends U ? P: never } & { [x: string]: never })[keyof T];
// If there is a single key, it must be a key of a number field
export function increment<
C extends Collection<any>,
E = C extends Collection<infer U> ? U : never,
K1 extends FilterKeysByType<E, number | undefined> = FilterKeysByType<E, number | undefined>,
V1 extends Exclude<E[K1], void> = Exclude<E[K1], void>
>(collection: C, entityId: string, path: K1 | [K1]): C
// If there are two keys, the first key, can be any type, but the second key must be the key of a number field
export function increment<
C extends Collection<any>,
E = C extends Collection<infer U> ? U : never,
K1 extends keyof E = keyof E,
V1 = Exclude<E[K1], void>,
K2 extends FilterKeysByType<V1, number| undefined> = FilterKeysByType<V1, number| undefined>,
V2 extends V1[K2]= V1[K2]
>(collection: C, entityId: string, path: [K1, K2]): C
// Usage
declare var c : Collection<Comment>;
interface Comment extends Entity{
name: string;
value: number;
optValue?: number;
subcomment: Comment;
optSubcomment?: Comment;
}
increment(c, "", "value"); //ok
increment(c, "", "optValue"); //ok
increment(c, "", "name");// error
increment(c, "", ["subcomment", "value"]); // ok
increment(c, "", ["subcomment", "optValue"]); // ok
increment(c, "", ["subcomment", "name"]); // error
increment(c, "", ["optSubcomment", "value"]); // ok
increment(c, "", ["optSubcomment", "optValue"]); // ok
increment(c, "", ["optSubcomment", "name"]); // error
关于typescript - 强制类型是接口(interface)的子类型并扩展编号,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49387333/