从Kotlin创建的Java SAM接口提供了ClassCastException

时间:2017-07-07 12:51:11

标签: kotlin

我有一个java方法:

addHandler(HttpServiceHandler handler)

HttpServiceHandler

interface HttpServiceHandler extends Consumer<HttpHandlerContext>

重点是避免在整个项目中进行Consumer<HttpHandlerContext>复制粘贴,因此它是一种类型别名。

在Java代码中,这可以正常工作:

addHandler({ context -> context.blah(); })

现在,在Kotlin中,我有这个生成处理程序的方法:

private companion object {
    fun newHandler(notimportant: Long): HttpServiceHandler {
        return HttpServiceHandler { context -> context.blah() }
    }
}

HttpServiceHandler {}很重要,如果我没有为lambda指定HttpServiceHandler,它就不会编译。

这个编译:

addHandler(newHandler(1L))

但是在运行时,抛出:

java.lang.ClassCastException: blah.BlahTest$Companion$newHandler$1 cannot be cast to kotlin.jvm.functions.Function1

at blah.BlahTest.test(BlahTest.kt:42)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:498)
at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47)
at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17)
at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:78)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:57)
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290)
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58)
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268)
at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
at org.junit.runner.JUnitCore.run(JUnitCore.java:137)
at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:68)
at com.intellij.rt.execution.junit.IdeaTestRunner$Repeater.startRunnerWithArgs(IdeaTestRunner.java:51)
at com.intellij.rt.execution.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:242)
at com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.java:70)

我无法弄清楚原因。请帮忙吗?

更新:如果我这样编写,则类强制转换异常消失了:

addHandler(
    object : HttpServiceHandler {
        override fun accept(c: HttpHandlerContext) {
            c.complete()
        }
    }
)

但是这样写的时候仍会抛出异常:

fun newHandler(blah: Long): HttpServiceHandler {
    return object : HttpServiceHandler {
        override fun accept(c: HttpHandlerContext) {
            c.complete()
        }
    }
}

addHandler(newHandler(1L))

我不明白为什么。

Update2: https://github.com/wilem82/testcases/tree/master/kotlinsam1的测试代码。遗憾的是,不能重现这个问题。

1 个答案:

答案 0 :(得分:1)

错误已经准确地说:

  

java.lang.ClassCastException:blah.BlahTest $ Companion $ newHandler $ 1无法强制转换为kotlin.jvm.functions.Function1

您尝试将Consumer<HttpHandlerContext>投射到Function1方法中的addHandler或某处,例如:

fun addHandler(handler: Consumer<HttpHandlerContext>) {
    val it: Function1<*, *> = handler as Function1<*, *>
    //                        ^
    // ClassCastException was thrown since it is not a Function1 
}

您应该使用kotlin中的java / function引用表达式中的方法引用表达式将SAM接口转换为另一个SAM接口,例如:

fun addHandler(handler: Consumer<HttpHandlerContext>) {
    val it: Function1<HttpHandlerContext, Unit> = handler::accept
    //     using function reference expression here     ---^
   //...
}

IF 你在kotlin中调用java方法addHandler,你不需要创建这样的桥接方法newHandler,你可以在kotlin中用lambda调用它,例如:

addHandler{context->
    context.blah()
    // additional works ...
}