如何在Kotlin中指定递归泛型参数?

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了如何在Kotlin中指定递归泛型参数?相关的知识,希望对你有一定的参考价值。

我只想从使用Log4j(v2.8.2)的Java中移植以下代码:

ConsoleAppender appender = ConsoleAppender.newBuilder().
            withName("ConsoleAppender").build();

问题在于newBuilder()方法,它被定义为log4j中的某种递归泛型:

@PluginBuilderFactory
public static <B extends Builder<B>> B newBuilder() {
    return new Builder<B>().asBuilder();
}

Java代码自动推断通用参数,而Kotlin则不然。有什么解决方案可以在Kotlin中调用这个方法吗?

我在Kotlin尝试过的代码:

val appender = ConsoleAppender.newBuilder().withName("ConsoleAppender").build()

它有以下错误:

错误:(90,48)Kotlin:类型推断失败:没有足够的信息来推断参数B的乐趣!> newBuilder():B!请明确说明。

当代码在粘贴时自动从Java转换时,它会设置一些stub newBuilder<B>(),其中B未定义,我不知道它应该是什么。

答案

Log4J似乎使用了一个构建器模式,在Kotlin中要求每个开放的非抽象类都有两个构建器:一个是泛型​​的,可以通过子类扩展,另一个是非泛型的,可以实例化。

既然你可能不想修改Log4J,我会考虑使用反射:

fun main(args: Array<String>) {
val builder = ConsoleAppender::class.java.getMethod("newBuilder").invoke(null) as ConsoleAppender.Builder<*>
val appender = builder.withName("ConsoleAppender").build()
println(appender.name)

}

请参阅我在discuss.kotlinlang.org的回答:

Recursive generic in Builder pattern

另一答案

我找到了等待如何为log4j2配置解决此问题,请参阅示例:

// create concrete type just to call method
// however, we should not use it, because of possible runtime casting issues
private class ConsoleBuilder : ConsoleAppender.Builder<ConsoleBuilder>()

private fun createAppender(configuration: Configuration): ConsoleAppender {
    // hide our temporary class immediately, to avoid runtime type casting issue
    return (ConsoleAppender.newBuilder<ConsoleBuilder>() as ConsoleAppender.Builder<ConsoleBuilder>).apply {
        withName("MyAppender")
        setConfiguration(configuration)
    }.build()
}

以上是关于如何在Kotlin中指定递归泛型参数?的主要内容,如果未能解决你的问题,请参考以下文章

如何在容器中指定模板化别名'泛型类型

如何解决本机 Java 或 Kotlin 中的错误“在依赖项的 AAR 元数据中指定的 minCompileSdk (31)”? [复制]

F#如何在递归可区分联合中指定类型限制

如何在 Kotlin 中获取泛型类型参数的类

微软统一。如何在构造函数中指定某个参数?

我可以在 XAML(.NET 4 之前)中指定泛型类型吗?