Kotlin - 是否可以推断泛型类型参数?

时间:2021-01-16 00:38:48

标签: kotlin generics

鉴于此代码:

// Can generic type parameter be inferred?
class Foo<T> {
    fun consume(p: T) {
        println("slurp $p")
    }
}

//region
abstract class ATestsOfFoo<U: Foo<T>, T>(private val foo: U) {

    fun test1() {
        foo.consume(someT())
    }

    abstract fun someT(): T
}

class TestsOfFoo(foo: Foo<String>): ATestsOfFoo<Foo<String>, String>(foo) {
    override fun someT(): String {
        return "Hello"
    }
}

val testsOfFoo = TestsOfFoo(Foo())
testsOfFoo.test1()
//endregion

我正在寻找一种从 T 推断 U 类型参数的方法。

换句话说,我希望能够在不重复 String 作为类型参数的情况下编写以下行:

...
// This is an example of what I'd like to be able to write, it does not work
class TestsOfFoo(foo: Foo<String>): ATestsOfFoo<Foo<String>>(foo) {
...

有没有办法做到这一点?

如果不是,是否存在禁止我们从 T 安全推断 U 的概念缺陷?

附: 我知道我们可以像这样简化 ATestsOfFoo

abstract class ATestsOfFoo<T>(private val foo: Foo<T>) {
...

但是我不是在寻找那个,从概念上讲 ATestsOfFooFoo<T> 进行操作,我想将该信息保留在类型参数中。

1 个答案:

答案 0 :(得分:1)

简而言之,不,Kotlin 不会像这样根据超级构造函数调用的参数类型推断类型。 Kotlin 的类型推断仅支持局部变量推断和函数签名类型推断。

您可以在 Kotlin 规范 (https://kotlinlang.org/spec/type-inference.html) 中阅读更多相关信息

我不确定这里是否存在概念上的缺陷,而且我在类型系统方面没有经验,但我知道这可以通过 TypeScript 的类型系统推断出来,但那是因为 TypeScript 的约束求解器和类型检查器可以使用条件类型和默认值泛型的类型,这不是 Kotlin 的语言特性。

interface Foo<T> {
  consume(p: T): void
}

interface FooImpl1<U extends Foo<T>, T = U extends Foo<infer X> ? X : never> {
  test(foo: U): T
}

class FooImpl implements FooImpl1<Foo<string>> {
  test(foo: Foo<string>): string { ... }
}

总结;使用 Kotlin 当前的类型系统,我们无法表达这一点。不幸的是,您只需要重复这些类型。

相关问题