在检查Kotlin协同程序的来源时,我注意到JDK 8 CompletableFuture
之间存在差异(标有**
)
public fun <T> future(
context: CoroutineContext = DefaultDispatcher,
start: CoroutineStart = CoroutineStart.DEFAULT,
block: suspend CoroutineScope.() -> T
): CompletableFuture<T> {
require(!start.isLazy) { "$start start is not supported" }
val newContext = newCoroutineContext(context)
val job = Job(newContext[Job])
val future = CompletableFutureCoroutine<T>(newContext + job)
job.cancelFutureOnCompletion(future)
** future.whenComplete { _, exception -> job.cancel(exception) } **
start(block, receiver=future, completion=future) // use the specified start strategy
return future
}
private class CompletableFutureCoroutine<T>(
override val context: CoroutineContext
) : CompletableFuture<T>(), Continuation<T>, CoroutineScope {
override val coroutineContext: CoroutineContext get() = context
override val isActive: Boolean get() = context[Job]!!.isActive
override fun resume(value: T) { complete(value) }
override fun resumeWithException(exception: Throwable) { completeExceptionally(exception) }
** doesn't override cancel which corresponds to interrupt task **
}
public fun <T> future(
context: CoroutineContext = DefaultDispatcher,
start: CoroutineStart = CoroutineStart.DEFAULT,
block: suspend CoroutineScope.() -> T
): ListenableFuture<T> {
require(!start.isLazy) { "$start start is not supported" }
val newContext = newCoroutineContext(context)
val job = Job(newContext[Job])
val future = ListenableFutureCoroutine<T>(newContext + job)
job.cancelFutureOnCompletion(future)
start(block, receiver=future, completion=future) // use the specified start strategy
return future
}
private class ListenableFutureCoroutine<T>(
override val context: CoroutineContext
) : AbstractFuture<T>(), Continuation<T>, CoroutineScope {
override val coroutineContext: CoroutineContext get() = context
override val isActive: Boolean get() = context[Job]!!.isActive
override fun resume(value: T) { set(value) }
override fun resumeWithException(exception: Throwable) { setException(exception) }
** override fun interruptTask() { context[Job]!!.cancel() } **
}
集成,虽然我认为类型几乎相同(当然除了ListenableFuture
不能直接完成,但我不明白为什么这在这里很重要)。这种差异背后是否有特定的原因?
答案 0 :(得分:6)
CompletableFuture.cancel是一种开放(可覆盖)的方法,但它不是设计用于覆盖。它的文档没有为取消调用提供任何保证,所以唯一能够证明CompletableFuture
被取消的未来证明(双关语)方法是在其上安装whenComplete
监听器。
例如,未来版本的JDK添加另一种方法来取消未在内部调用cancel
的未来,这是完全合法的。这样的改变不会违反任何CompletableFuture
合同。
将此与AbstractFuture.interruptTask上的文档进行比较。此方法明确设计用于覆盖,其文档保证调用它的条件。因此,我们可以为ListenableFuture
构建器提供稍微更有效的实现,避免创建lambda来在其上安装取消侦听器。