4

我在打字稿中有以下通用类

type UserId = number
type Primitive = string | number | boolean
class ColumnValue<T, S extends Primitive> {
    constructor(public columnName: String, public value: S) { }
}
abstract class Column<T> {
    constructor(public columnName: String) { }
    public set<S extends Primitive>(value: T): ColumnValue<T, S> {
        return new ColumnValue(this.columnName, this.getValue(value))
    }
    public abstract getValue<S extends Primitive>(value: T): S
}
let id = new class extends Column<UserId> {
    constructor() { super("id") }
    public getValue(value: UserId): number {
        return value
    }
}()

但我不知道为什么会出现此错误类“(匿名类)”错误地扩展了基类“列”。属性“getValue”的类型不兼容。类型 '(value: number) => number' 不可分配给类型 '(value: number) => S'。类型“数字”不可分配给类型“S”

4

2 回答 2

3

Columngetter 和 setterS上不一定是同一类型,因此您应该将类​​型参数移至其父类:Column<T, S extends Primitive>.

type UserId = number
type Primitive = string | number | boolean
class ColumnValue<T, S extends Primitive> {
    constructor(public columnName: String, public value: S) { }
}
abstract class Column<T, S extends Primitive> {
    constructor(public columnName: String) { }
    public set(value: T): ColumnValue<T, S> {
        return new ColumnValue(this.columnName, this.getValue(value))
    }
    public abstract getValue(value: T): S
}
let id = new class extends Column<UserId, number> {
    constructor() { super("id") }
    public getValue(value: UserId): number {
        return value
    }
}()

上面的版本至少没有错误。

我知道您可能想S从您与 setter 一起使用的任何类型中推断出,但Column在实例化时必须具有明确定义的类型,这意味着您要么在调用构造函数时显式(即new Column<UserId, number>(...)),要么在构造函数中添加一个S参数所以S可以从中推断(如new Column<UserId>('id', 123)

于 2017-09-24T11:03:23.317 回答
1

getValue使用的是 generic S,因此,继承的实现也必须使用S

let id = new class extends Column<UserId> {
    constructor() { super("id") }
    public getValue<S extends Primative>(value: UserId): S {
        return <S>value
    }
}()

如果你S上课,你的功能可以缩小到number.

abstract class Column<T, S extends Primative> {
    constructor(public columnName: String) { }
    public set(value: T): ColumnValue<T, S> {
        return new ColumnValue(this.columnName, this.getValue(value))
    }
    public abstract getValue(value: T): S
}

let id = new class extends Column<UserId, UserId> {
    constructor() { super("id") }
    public getValue(value: UserId): number {
        return value
    }
}()
于 2017-09-24T10:59:18.570 回答