我有一个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的测试代码。遗憾的是,不能重现这个问题。
答案 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 ...
}