Kotlin 协程 CoroutineScope

协程定义:

19年官方是这样说的:协程是轻量级的线程,协程就是 Kotlin 提供的一套线程封装的 API;

现在官方是这样说的:协程是一种并发设计模式;

协程作用:

1.处理耗时任务;

2.保证主线程的安全;

3.简化异步执行的代码,解决并发问题,让「协作式多任务」实现起来更加方便。

使用协程,同样可以像 Rx 那样有效地消除回调地狱,不过无论是设计理念,还是代码风格,两者是有很大区别的,协程在写法上和普通的顺序代码类似。

协程特点:

轻量:您可以在单个线程上运行多个协程,因为协程支持挂起,不会使正在运行协程的线程阻塞。挂起比阻塞节省内存,且支持多个并行操作。

内存泄漏更少:使用结构化并发机制在一个作用域内执行多项操作。

内置取消支持:取消功能会自动通过正在运行的协程层次结构传播。

协程使用: 

1.lauch

lauch用于在协程作用域中异步启动一个新的协程,调用该方法不会阻塞线程。

CoroutineScope(Dispatchers.IO).launch {
	// 启动一个非阻塞线程的协程
}

2.suspend

suspend是协程的关键字,每一个被suspend修饰的方法都必须在另一个suspend函数或者Coroutine协程程序中进行调用。 

3.coroutineScope

coroutineScope是一个挂起函数,每一个被suspend修饰的方法都必须在另一个suspend函数或者Coroutine协程程序中进行调用。

CoroutineScope(Dispatchers.IO).launch {
	coroutineScope {
		// 启动一个非阻塞线程的协程
	}
}

runBlocking {
	coroutineScope {
		// 启动一个非阻塞线程的协程
	}
}

4.runBlocking

runBlocking会阻塞当前线程,而coroutineScope不会阻塞所在的线程,它会挂起所在的协程直至其内部任务(包括子协程)执行完成。

runBlocking {
	// 启动一个阻塞线程的协程
}

5.dispatcher

dispatcher 协程调度器,可以控制协程代码块在UI线程还是子线程中执行;

6.async

1)在概念上,async 就类似于 launch。它启动了一个单独的协程与其它所有的协程一起并发的工作。不同之处在于 launch 返回一个 Job 并且不附带任何结果值,而 async 返回一个 Deferred接口指向的对象,使用 Deferred.await()在一个延期的值上得到它的最终结果,同时Deferred 也是一个 Job,所以如果需要的话,可以使用Deferred.cancel()取消它。

CoroutineScope(Dispatchers.IO).launch {
	val api1Deferred = async { api1() }
	val api2Deferred = async { api2() }
	val api3Deferred = async { api3() }
	println("api*****1")
	val result1 = api1Deferred.await()
    println("api*****2")
}

suspend fun api1(): String {
	delay(2500) // 模拟耗时操作
	println("api1")
	return "api1"
}

suspend fun api2(): String {
	delay(2000) // 模拟耗时操作
	println("api2")
	return "api2"
}

suspend fun api3(): String {
	delay(1500) // 模拟耗时操作
	println("api3")
	return "api3"
}

输出结果: 

结论:

async启动的所有协程是一起并发工作的,async不是挂起函数,所以不会挂起launch创建的协程,所以先输出api*****1,再api3、api2、api1,但执行到api1Deferred.await()时,会挂起协程,等待async { api1() }执行完返回结果后,再执行协程后续的函数,所以最后输出api*****2,因为await()是挂起函数。

public suspend fun await(): T

2)案例:有4个耗时方法,方法名api1,api2,api3,api4;要求方法api1先执行,返回String "api1",然后将结果"api1"作为参数,并发执行方法api2和方法api3,由于api3 delay(1500),api2 delay(2000),所以api3会先执行完并输出结果,然后api2再执行完并输出结果,最后将结果"api2",api3"作为参数,执行方法api4。

CoroutineScope(Dispatchers.IO).launch {
	val api1Deferred = async { api1() }
	println("api*****1")
	val result1 = api1Deferred.await()
	println("api1:$result1") // 等待 api1 方法执行完成并输出结果
	val api2Deferred = async { api2(result1) }
	val api3Deferred = async { api3(result1) }
	val result2 = api2Deferred.await()
	println("api2:$result2")
	val result3 = api3Deferred.await()
	println("api3:$result3")
	println(api4(result2, result3))
}

suspend fun api1(): String {
	delay(3000) // 模拟耗时操作
	println("api1")
	return "api1"
}

suspend fun api2(v: String): String {
	delay(2000) // 模拟耗时操作
	println("api2")
	return "api2:$v"
}

suspend fun api3(v: String): String {
	delay(1500) // 模拟耗时操作
	println("api3")
	return "api3:$v"
}

suspend fun api4(v: String, v2: String): String {
	delay(500) // 模拟耗时操作
	return "api4:$v$v2"
}

输出结果:  

注:async是并行的,如果使用await()的话,await()是挂起函数,会挂起协程,等待async { api1() }执行完返回结果后,再执行协程后续的函数。

7. withContext

withContext 与 async 都可以返回耗时任务的执行结果。多个 withContext 任务是串行(顺序执行)的, 且withContext 可直接返回耗时任务的结果。 而多个 async 任务是并行的。

public suspend fun <T> withContext()

因为withContext()是挂起函数,执行后,会挂起协程,等待withContext内部函数执行完后,再执行withContext函数后面的函数。

CoroutineScope(Dispatchers.IO).launch {
	val result1 = withContext(Dispatchers.IO) {
		api1()
	}
	println("api*****1")
	println("api1:$result1") // 等待 api1 方法执行完成并输出结果
	val result2 = withContext(Dispatchers.IO) {
		api2(result1)
	}
	println("api2:$result2")
	val result3 = withContext(Dispatchers.IO) {
		api3(result1)
	}
	println("api3:$result3")
	val result4 = withContext(Dispatchers.IO) {
		api4(result2, result3)
	}
	println(result4)
}

suspend fun api1(): String {
	delay(3000) // 模拟耗时操作
	println("api1")
	return "api1"
}

suspend fun api2(v: String): String {
	delay(2000) // 模拟耗时操作
	println("api2")
	return "api2:$v"
}

suspend fun api3(v: String): String {
	delay(1500) // 模拟耗时操作
	println("api3")
	return "api3:$v"
}

suspend fun api4(v: String, v2: String): String {
	delay(500) // 模拟耗时操作
	return "api4:$v$v2"
}

输出结果:   

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

sziitjin

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值