Kotlin 协程Channel 通道 ④ ( Channel 通道的热数据流属性 | Channel 通道关闭过程 | Channel 通道关闭代码示例 )

Posted 韩曙亮

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Kotlin 协程Channel 通道 ④ ( Channel 通道的热数据流属性 | Channel 通道关闭过程 | Channel 通道关闭代码示例 )相关的知识,希望对你有一定的参考价值。

文章目录





一、Channel 通道的热数据流属性



调用 CoroutineScope#produce 函数 构造的 生产者协程 , 以及 调用 CoroutineScope#actor 函数 构造的 消费者协程 ,

如果上述 生产者协程 和 消费者协程 执行完毕 , 则 对应的 Channel 通道 也会进行关闭 ,

因此 , Channel 通道 被称为 热数据流 ;

与 Channel 通道 热数据流 相对的是 Flow 异步流 的冷数据流 特征 ;





二、Channel 通道关闭过程



如果调用 Channel 通道的 Channel#close 函数 , 该 Channel 通道会 停止接收 新的数据 ,

此时调用 Channel#isClosedForSend 函数 , 会返回 true ;

Channel 通道存在缓冲区 , 通道不接收新的元素 , 但是 缓冲区已存储的元素需要被处理完毕 , 然后才能关闭通道 ,

当 Channel 通道 缓冲区 所有的元素处理完毕 , 调用 Channel#isClosedForReceive 函数 会返回 true ;





三、Channel 通道关闭代码示例



在下面的代码中 , Channel 通道缓冲区大小为 3 , 数据生产者 一次性将 3 个数据全部发送出去 , 但是 数据消费者 每秒只能消费一个数据 , 需要 3 秒才能将数据处理完毕 ;

在发送完数据后 , 调用 Channel#close 函数 , 关闭通道 , 此时 channel.isClosedForSend 为 true , 但是 channel.isClosedForReceive 为 false ;

在接收数据完毕后 , channel.isClosedForReceive 才变为 true , Channel 通道正式关闭 ;


代码示例 :

package kim.hsl.coroutine

import android.os.Bundle
import androidx.appcompat.app.AppCompatActivity
import kotlinx.coroutines.*
import kotlinx.coroutines.channels.Channel

class MainActivity : AppCompatActivity() 
    override fun onCreate(savedInstanceState: Bundle?) 
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)

        runBlocking 
            // Channel 通道, 传递 Int 类型数据
            val channel = Channel<Int>(3)

            // 数据生产者协程
            val producer = GlobalScope.launch 
                for (i in 0..2) 
                    channel.send(i)
                    println("向通道中发送数据 $i, channel.isClosedForSend : $channel.isClosedForSend, channel.isClosedForReceive : $channel.isClosedForReceive")
                
                // 关闭通道
                channel.close()
                println("数据发送完毕, channel.isClosedForSend : $channel.isClosedForSend, channel.isClosedForReceive : $channel.isClosedForReceive")
            

            // 数据消费者协程
            val consumer = GlobalScope.launch 
                for(num in channel) 
                    delay(1000)
                    println("从通道中接收数据 $num, channel.isClosedForSend : $channel.isClosedForSend, channel.isClosedForReceive : $channel.isClosedForReceive")
                
                println("数据接收完毕, channel.isClosedForSend : $channel.isClosedForSend, channel.isClosedForReceive : $channel.isClosedForReceive")

            

            // 等待两个协程执行完毕
            joinAll(producer, consumer)
        
    

执行结果 :

2022-12-28 11:07:06.509 I/System.out: 向通道中发送数据 0, channel.isClosedForSend : false, channel.isClosedForReceive : false
2022-12-28 11:07:06.509 I/System.out: 向通道中发送数据 1, channel.isClosedForSend : false, channel.isClosedForReceive : false
2022-12-28 11:07:06.509 I/System.out: 向通道中发送数据 2, channel.isClosedForSend : false, channel.isClosedForReceive : false
2022-12-28 11:07:06.510 I/System.out: 数据发送完毕, channel.isClosedForSend : true, channel.isClosedForReceive : false
2022-12-28 11:07:07.516 I/System.out: 从通道中接收数据 0, channel.isClosedForSend : true, channel.isClosedForReceive : false
2022-12-28 11:07:08.527 I/System.out: 从通道中接收数据 1, channel.isClosedForSend : true, channel.isClosedForReceive : false
2022-12-28 11:07:09.540 I/System.out: 从通道中接收数据 2, channel.isClosedForSend : true, channel.isClosedForReceive : true
2022-12-28 11:07:09.540 I/System.out: 数据接收完毕, channel.isClosedForSend : true, channel.isClosedForReceive : true

以上是关于Kotlin 协程Channel 通道 ④ ( Channel 通道的热数据流属性 | Channel 通道关闭过程 | Channel 通道关闭代码示例 )的主要内容,如果未能解决你的问题,请参考以下文章

Kotlin 协程Channel 通道 ② ( Channel 通道容量 | Channel 通道迭代 | 使用 iterator 迭代器进行迭代 | 使用 for in 循环进行迭代 )

Kotlin 协程Channel 通道 ① ( Channel#send 发送数据 | Channel#receive 接收数据 )

Kotlin 协程Channel 通道 ① ( Channel#send 发送数据 | Channel#receive 接收数据 )

Kotlin 协程Channel 通道 ⑤ ( BroadcastChannel 广播通道 | 代码示例 )

Kotlin 协程协程中的多路复用技术 ① ( 多路复用技术 | await 协程多路复用 | Channel 通道多路复用 )

Kotlin 协程Channel 通道 ③ ( CoroutineScope#produce 构造生产者协程 | CoroutineScope#actor 构造消费者协程 )