我正在学习如何使用CompletableFuture使协程与Java库一起使用。下面是我的代码:
// x invokes y invokes z invokes Java client
suspend fun x(i: Int, client: FakeJavaClient): Int {
fun z(k: Int): Int {
println("z: $k")
return client.query(k).get()
}
tailrec // with 'tailrec', the code never terminates
suspend fun y(j: Int): Int {
val ret = z(j)
if (ret > 10) {
return ret
}
return y(j + 1)
}
return y(i)
}
fun main() {
runBlocking {
launch(Dispatchers.IO) {
FakeJavaClient().use { x(0, it) }
}
}
println("Done")
}
class FakeJavaClient : AutoCloseable {
private val executor = Executors.newFixedThreadPool(10)
fun query(i: Int): CompletableFuture<Int> {
val f = CompletableFuture<Int>()
executor.submit {
Thread.sleep(1000)
f.complete(i * 2)
}
return f
}
override fun close() {
executor.shutdown()
executor.awaitTermination(10, TimeUnit.SECONDS)
}
}
如果我将
tailrec
修饰符添加到y
函数中,则代码输出如下,并且永远不会结束:z: 0
z: 0
z: 0
z: 0
z: 0
...
如果我删除
tailrec
上的y
,则代码的行为符合我的期望z: 0
z: 1
z: 2
z: 3
z: 4
z: 5
z: 6
Done
有人可以帮助我了解这里发生的事情吗?
最佳答案
这看起来像一个已知问题:Wrong code generated for a local tailrec suspend function。
关于Kotlin:tailrec使暂停乐趣永无止境,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/60647842/