我可以通过在 Java 代码中使用 Kotlin 的协程来利用它们吗?

2022-09-02 12:51:07

我的目标是什么?

我的目标是能够从 Java 使用 Kotlin 的 Coroutine 系统。我希望能够在给定的时间内暂停执行,然后在给定的时间量过后在该位置恢复。从Java中,我希望能够执行允许在不以异步方式暂停执行过程中的任务,例如:

//example 1
someLogic();
pause(3000L); //3 seconds
someMoreLogic();

//example 2
while(true) {
    someContinuedLogic();
    pause(10000L); //10 seconds
}

我的问题是什么?

正如预期的那样,我能够从Kotlin完美地执行协程,但是当涉及到Java时,它变得棘手,因为代码的Java部分一次执行整个块而没有任何暂停,而Kotlin块正确地暂停1秒,然后是4秒。

我的问题是什么?

是否有可能使用 Kotlin 作为 Java 中协程的骨干?如果是这样,我做错了什么?您可以在下面找到源代码,显示我如何尝试在Java中使用Kotlin的协程。

KtScript 类

abstract class KtScript {

    abstract fun execute()

    fun <T> async(block: suspend () -> T): CompletableFuture<T> {
        val future = CompletableFuture<T>()
        block.startCoroutine(completion = object : Continuation<T> {
            override fun resume(value: T) {
                future.complete(value)
            }
            override fun resumeWithException(exception: Throwable) {
                future.completeExceptionally(exception)
            }
        })
        return future
    }

    suspend fun <T> await(f: CompletableFuture<T>): T =
            suspendCoroutine { c: Continuation<T> ->
                f.whenComplete { result, exception ->
                    if (exception == null)
                        c.resume(result)
                    else
                        c.resumeWithException(exception)
                }
            }

    fun pause(ms: Long): CompletableFuture<*> {
        //todo - a better pausing system (this is just temporary!)
        return CompletableFuture.runAsync {
            val currentMs = System.currentTimeMillis()
            while (System.currentTimeMillis() - currentMs < ms) {
                /* do nothing */
            }
        }
    }

}

Kotlin 执行代码

fun main(args: Array<String>) {
    ScriptTestKotlin().execute()
}

class ScriptTestKotlin : KtScript() {
    override fun execute() {
        println("Executing Kotlin script from Kotlin...")
        val future = async {
            await(pause(1000L))
            println("   1 second passed...")
            await(pause(4000L))
            println("   5 seconds passed...")
        }
        future.get() //wait for asynchronous task to finish
        println("Finished!")
    }
}

Kotlin 执行结果

Executing Kotlin script from Kotlin...
   1 second passed...
   5 seconds passed...
Finished!

Java 执行代码

public class ScriptTestJava extends KtScript {

    public static void main(String[] args) {
        new ScriptTestJava().execute();
    }

    @Override
    public void execute() {
        System.out.println("Executing Kotlin script from Java...");
        CompletableFuture<?> future = async(continuation -> {
            await(pause(1000L), continuation);
            System.out.println("    1 second passed...");
            await(pause(4000L), continuation);
            System.out.println("    5 seconds passed...");
            return continuation;
        });
        try {
            future.get(); //wait for asynchronous task to finish
        } catch(Exception e) {
            e.printStackTrace();
        }
        System.out.println("Finished!");
    }
}

Java 执行结果

Executing Kotlin script from Java...
    1 second passed...
    5 seconds passed...
Finished!

^^^ 不幸的是,在Java中跳过了暂停。^^^


答案 1

Kotlin 协程是通过编译器对代码的转换来实现的,这显然只能由 来完成。kotlinc

所以,不,Java不能使用Kotlin的协程机制,因为它是一个编译时功能。


答案 2

推荐