目录:
一. 协程的基本概念
协程就像非常轻量级的线程。线程是由系统调度的,线程切换或线程阻塞的开销都比较大。而协程依赖于线程,但是协程挂起时不需要阻塞线程,几乎是无代价的,协程是由开发者控制的。
二. 从异步编程开始
我们先从一个例子说起,发送一个带有认证的 post 请求,需要以下三个步骤,首先客户端向服务其发送一个得到token的请求,然后构造一个 Post 请求,最后将 Post 请求发出去。这三个请求都是耗时操作,而且请求和请求之间有着依赖的关系。fun requestToken(): Token {
delay(500L) // 模拟请求过程
return token
}
fun createPost(token: Token, item: Item): Post {
delay(500L) // 模拟构造过程
return post
}
fun processPost(post: Post) {
delay(500L) // 模拟请求过程
}
方法一:使用回调的方式
操作2依赖于操作1,所以把操作2作为回调放在操作1的参数内,由操作1决定回调时机。
fun requestTokenAsync(cb: (Token) -> Unit) { ... }
fun createPostAsync(token: Token, item: Item, cb: (Post) -> Unit) { ... }
fun processPost(post: Post) { ... }
fun postItem(item: Item) {
requestTokenAsync { token ->
createPostAsync(token, item) { post ->
processPost(post)
}
}
}
这种多层嵌套的方式比较复杂,而且不方便处理异常情况。
方法二:CompletableFuture
Java 8 引入的 CompletableFuture 可以将多个任务串联起来,可以避免多层嵌套的问题。
可以简单看一下API,具体的使用方法参考文章:
CompletableFuture 使用详解
方法 | 作用 |
---|---|
runAsync | 创建一个异步操作,不支持返回值 |
supplyAsync | 创建一个异步操作,支持返回值 |
whenComplete | 计算结果完成的回调方法 |
exceptionally | 计算结果出现异常的回调方法 |
thenApply | 当一个线程依赖另一个线程时,可以使用 thenApply 方法来把这两个线程串行化。 |
handle | 与thenApply相似,handle还可以处理异常任务 |
thenAccept | 与thenApply相似,但是没有返回值 |
thenRun | 与thenAccept相似,但是得不到上面任务的处理结果 | <