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

问题描述 投票:4回答:2

我只想从使用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未定义,我不知道它应该是什么。

generics kotlin log4j
2个回答
2
投票

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


0
投票

我找到了等待如何为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()
}
© www.soinside.com 2019 - 2024. All rights reserved.