Kotlin(十一)设计模式-行为型模式(观察者)

koitlin 行为模式---观察者模式

简单来说,需要满足的两件事:

  1. 订阅者,添加或者删除 观察者的监听
  2. 发布者状态改变的时候,通知观察者,观察者执行相应的响应逻辑

java 自身标准提供了java.util.Observable 类和java.util.Observer接口来帮助实现观察者模式。实现一个动态更新价格的例子;

import java.util.*

class StockUpdate: Observable() {
    val observers = mutableSetOf<Observer>();
    fun setStockChanched(price:Int){
        this.observers.forEach { it.update(this,price) }
    }
}

class StockDisplay:Observer{
    override fun update(o: Observable?, price: Any?) {
        if(o is StockUpdate){
            println("The latest stock price is $price")
        }
    }
}

fun main() {
    val su = StockUpdate()
    val sd = StockDisplay()
    su.observers.add(sd)
    su.setStockChanched(200)
}  
Observable

Kotlin 的标准库引入了可被观察者委托属性,也可以利用来实现同样的场景

import kotlin.properties.Delegates

interface StockUpdateListener{
    fun onRise(price:Int)
    fun onFall(price:Int)
}

class StockDisplay2:StockUpdateListener{
    override fun onRise(price: Int) {
        println("The latest stock prise has risen to $price")
    }

    override fun onFall(price: Int) {
        println("The latest stock prise has fall to $price")
    }

}

class StockUpdate2{
    var listeners = mutableSetOf<StockUpdateListener>()

    var price:Int by Delegates.observable(0){_,old,new->
        listeners.forEach {
            if(new > old) it.onRise(price) else it.onFall(price)
        }

    }
}

fun main() {
    val su = StockUpdate2()
    val sd = StockDisplay2()
    su.listeners.add(sd)
    su.price = 100
    su.price = 98

这里我们添加了功能,价格上涨和下跌观察。java.util.Observer 的接口类只能复写update方法来写相应逻辑,也就是说,如果存在多种不同的逻辑响应,我们也必须通过在改方法中进行区分实现。这回让订阅者显得臃肿。如果我们把发布者的事件推送看成一个第三方服务,那么他提供的API接口只有一个update,API调用者必须承担更多的职能。

使用Delegates.observable() 的方案更加灵活。提供了三个参数,一次代表:委托属性的元数据KProperty对象,旧值以及新值。通过额外定义一个StockUpdateListener接口,上涨和下跌的不同逻辑响应封装成了接口方法,从而,StockDisplay 中实现该接口的onRise和onFall 方法实现了解耦

Vetoable

我们有时候不希望监控的值随意更改,可能希望对某些改值的情况进行否决。Kotlin标准库除了observable这个委托属性之外,还提供了另外一个属性,vetoable,提供了功能,在被赋值前进行截获,判断是否接受。

var value:Int by Delegates.vetoable(0){
    prop,old,new->
    new > 0
}
>>>>
    value = 1
    println(value)
    value = -1
    println(value)

>>>>
1
1

初始值0,视图value更改为-1的时候结果仍未旧值1

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容