Kotlin-channel的基本使用

本文介绍了Kotlin的Channel概念,包括其源代码解析、构建方法以及如何使用Channel实现View防止重复点击。Channel分为有缓冲区和无缓冲区,通过produce()和actor()函数创建,其中produce()返回ReceiveChannel,actor()返回SendChannel。通过示例展示了Channel在防止点击事件重复触发中的应用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Channel的源代码如下:

public interface Channel<E> : SendChannel<E>, ReceiveChannel<E> {
   ...
}
  • Channel的父类有发送消息的SendChannel和接受消息的ReceiveChannel,Channel分为有缓冲区和无缓冲区,无缓冲的通道在发送者和接收者相遇时传输元素。如果发送先被调用,则它将被挂起直到接收被调用, 如果接收先被调用,它将被挂起直到发送被调用。Channel() 工厂函数与 produce和actor 建造器通过一个可选的参数 capacity 来指定 缓冲区大小 。缓冲允许发送者在被挂起前发送多个元素, 就像 BlockingQueue 有指定的容量一样,当缓冲区被占满的时候将会引起阻塞。

Channel的构建

  • GlobalScope 中提供2个函数 produce() 和 actor()

    • produce()函数返回值是一个ReceiveChannel对象,最后一个参数是一个继承SendChannel的对象,使用代码如下:
      val data = GlobalScope.produce<String> {
             send("a")
         }.receive()
    
    • actor()函数的最后一个参数是一个继承ActorScope类的对象,ActorScope类继承了ReceiveChannel类,用来接受处理函数返回一个SendChannel对象发送的消息,使用代码如下:
     GlobalScope.actor<String> {
             val data = receive()
         }.send("a")
    

    produce()和actor()第二个参数capacity 默认值是0,表示构建的channel是无缓冲区的,若重新赋值为大约0,则构建的channel是有缓冲区的。

    • 使用标准库中的iterator()函数构建一个相似的管道,使用 iterator 替换 produce、yield 替换 send、next 替换 receive、 Iterator 替换 ReceiveChannel 来摆脱协程作用域,你将不再需要 runBlocking。代码如下:
    iterator<String> {
         yield("ss")
     }.next()
    
    • 直接初始化,send()和receive()函数是挂起函数,只能在协程或者其他挂起函数中调用。代码如下:
     GlobalScope.launch {
         val channel = Channel<String>()
         channel.send("11")
         val data = channel.receive()
     }
    
    **Channel()参数capacity 默认值是0,表示构建的channel是无缓冲区的,若重新赋值为大约0,则构建的channel是有缓冲区的。**
    
  • 计时器Channel-- ticker (),使用代码如下:

 GlobalScope.launch(Dispatchers.Main) {
            val tickerChannle = ticker(1,
             initialDelayMillis = 0)
            tickerChannle.receive()
            tickerChannle.cancel()
        }

使用Channel 实现View 防止重复点击

fun View.setOnceClick(block: suspend () -> Unit) {
   val action = GlobalScope.actor<Unit> {
       for (event in channel){ 
       block()
        delay(2000)//延迟2s
}
   }
   setOnClickListener {
       action.offer(Unit)
   }
}
  • 在View 类中扩展一个函数setOnceClick(),在该函数中调用View的点击事件函数, GlobalScope.actor()函数返回一个SendChannel 对象,在点击事件中调用offer()函数,发送消息。
  • actor()函数的最后一个参数是一个继承ActorScope类的对象,ActorScope类继承了ReceiveChannel类,用来接受处理函数返回一个SendChannel对象发送的消息
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值