鉴于此代码:
// 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>) {
...
但是我不是在寻找那个,从概念上讲 ATestsOfFoo
对 Foo<T>
进行操作,我想将该信息保留在类型参数中。
答案 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 当前的类型系统,我们无法表达这一点。不幸的是,您只需要重复这些类型。