我想知道在TS中是否可以强制使用通用属性的类型。我只想允许传递带有“字符串”属性的对象类型的泛型。例如,如果所传递的通用接口包含数字或符号属性,则会引发错误。
以下是我尝试并评论过的行为的POC:
class Test<T extends {[key: string]: any}>{
private data: T;
public getValue<K extends keyof T>(key: K): T[K] {
return this.data[key];
}
}
// the property is a string = ok
const okay = new Test<{ "aString": string }>();
// the property is a number = should raise an error
const shouldFail = new Test<{ 0: string }>();
答案 0 :(得分:1)
如果对象具有字符串索引,我们也可以按数字索引该对象,因此编译器没有理由抛出错误数字键。这是设计使然。
declare let skeys: { [key: string]: number }
let v1 = skeys[0] // number
let v2 = skeys["0"] // number
declare let nkeys: { [key: number]: number }
let v3 = nkeys[0] // number
let v4 = nkeys["0"] // error
declare let snkeys: {
[key: number]: number;
[key: string]: string | number // The string index has to contain any value reuned by the number index
}
let v5 = snkeys[0] // number
let v6 = snkeys["0"] // string| number
如果对象包含任何非字符串键,则可以使用条件类型强制错误。该错误将不会很严重,但它可以读取并且可以完成工作:
class Test<T extends { [key: string]: any } & (keyof T extends string ? {} : "T must obnly have string keys") >{
private data!: T;
public getValue<K extends keyof T>(key: K): T[K] {
return this.data[key];
}
}
// the property is a string = ok
const okay = new Test<{ "aString": string }>();
// Error: Type '{ 0: string; }' does not satisfy the constraint '{ [key: string]: any; } & "T must only have string keys"'.
const shouldFail = new Test<{ 0: string }>();
注意
如果您对T
的值没有任何其他限制,那么简单的object
类型也可以使用
class Test<T extends object & (keyof T extends string ? {} : "T must only have string keys") >{ }