launch和async都是CoroutineScope
的扩展函数,用于启动一个新的协程。
核心区别
launch() | async() | |
返回值 | 返回Job表示任务状态 | 返回Deferred<T> 可获取结果T |
用途 |
执行不需要结果的任务 用于启动一个新的协程 |
执行需要结果的任务 用于启动一个并发任务 |
异常处理 | 内部发生异常会立即传播,会导致协程取消 | 异常暂存,直到调用await() 时抛出 |
使用场景 |
网络请求、IO操作、UI更新 适合不需要返回值的任务(如日志上传、UI更新、IO操作) |
合并数据、同时请求多个接口 适合需要并发执行多个任务并汇总结果的场景(如并行请求多个接口) |
具体事例
1. launch() 不关心结果
//启动任务,不等待结果
val launchA = launch{ Task() }
val launchB = launch{ Task() }
//继续其他任务
....
2. async()需要等待其他结果
//启动任务,需要等待结果
val asyncA = async{ TaskA() }
val asyncB = async{ TaskB() }
val a = asyncA.await()
val b = asyncB.await()
//两个任务并行执行 需要等待a b 都执行完以后在执行后续任务
val asyncC = async{ TaskC() }
重点细节
1. async必须调用await获取结果
val async = async { Task() }
val res = async.await()//如果不嗲用await结果可能会被忽略,且异常被静默吞掉!
2.异常处理的不同
launch()异常会立即取消父协程
val la = launch{ throw RuntimeException("Exception error!")} //会导致父协程取消
async()
异常暂存,调用 await()
时才抛出:
val async= async{ throw RuntimeException("Exception error!")} //会导致父协程取消
delay(1000)
async.wait()//此时才会抛出异常
3.async()+awaitAll() 处理并发任务
val asyncA = async{ TaskA() }
val asyncB = async{ TaskB() }
val asyncC = async{ TaskC() }
val result = awaitAll(asyncA,asyncB,asyncC)//等所有结果
开发中launch()
和 async()
的选择使用场景
launch()
:
- 不需要任务返回结果。
- 独立执行的任务(比如埋点、日志上传等)。
- 需要更简单的异常处理(立即失败)。
async()
:
- 需要并发执行多个任务并汇总结果。
- 需要处理任务间的依赖(如任务A的结果传给任务B)。
- 需要更灵活地控制异常(如延迟处理)。