Android潜修者Android知识Android技术知识

设计模式之观察者模式

2016-11-06  本文已影响93人  cgzysan

观察者模式

Rxjava中运用到了观察者模式,那什么是观察者模式呢,现在来学习一下。正所谓观察,就是看,细察事物的现象、动向。在这里看来就是监视目标的动作,然后自己作出相应的回应。

先写一个简单的观察者

再写一个被观察者

写一个线程类

既然要观察监视,就要时刻关注被观察者的状态,以便能在其发生改变的时候及时的更新自己的状态,那么我们就开一条子线程一直在其中循环判断嘛,这不就行了么。

public class WatchThread extends Thread {
    // 被观察者
    private RealObservable mRealObservable;
    // 观察者
    private RealObserver mReObserver;
    // 具体的监视事件
    private String mType;
    
    WatchThread(RealObservable realObservable ,RealObserver realObserver ,String type){
        this.mRealObservable = realObservable;
        this.mReObserver = realObserver;
        this.mType = type;
    }

    @Override
    public void run() {
        while (true) {
            if (this.mType.equals("eat")) {
                if (this.mRealObservable.isEat()){
                    this.mReObserver.update("吃饭。。。");
                    //重置状态,以便继续继续监视更新状态
                    this.mRealObservable.setEat(false);
                }
            } else {
                if (this.mRealObservable.isSleep()){
                    this.mReObserver.update("睡觉了。。。");
                    //重置状态,以便继续继续监视更新状态
                    this.mRealObservable.setSleep(false);
                }
            }
        }
    }
}

开启线程

那么现在,万事俱备,只欠东风。在主线程中,创建子线程,并开启线程,获取打印结果。

public static void main(String[] args) throws InterruptedException {
    //定义出观察者和被观察者
    RealObservable realObservable = new RealObservable();
    RealObserver realObserver= new RealObserver();
    
    //创建监视吃饭线程,并开启
    WatchThread eatWatch = new WatchThread(realObservable, realObserver, "eat");
    eatWatch.start();
    
    //创建监视睡觉线程,并开启
    WatchThread sleepWatch = new WatchThread(realObservable, realObserver, "sleep");
    sleepWatch.start();
    
    //主线程休眠1.5s,确保子线程开启完成
    Thread.sleep(1500);
    //改变被观察者状态
    realObservable.eat();
    
    Thread.sleep(1000);
    realObservable.sleep();
}

打印结果:</br>
我吃饭了。。。</br>
观察到对方动作,开始汇报</br>
老板,有情况。。。目标在吃饭。。。</br>
汇报完毕</br>
我睡觉了。。。</br>
观察到对方动作,开始汇报</br>
老板,有情况。。。目标在睡觉了。。。</br>
汇报完毕</br>

观察者的另一层含义

那么,这样并不算完,注意到没有,我们在子线程开一个无限死循环while(true)作监听,这样的成本太高。那应该如何进行修改呢,这里换一个方向想,我们需要的是什么,我们需要的是如果被观察者的状态发生了改变,或者说触发了什么事件,我们能够及时的获得这个情报,我们也只需要这个情报,想想看过的谍战片,一直派人监视目标是不是很累,并且需要大量的资源,就像这里的while(true)死循环一样,而有的人就买通目标身边的亲信,让亲信给自己送情报,一旦目标有什么动作就通知自己,这样是不是降低了很多成本,当然买通亲信也是要成本的咯。这也就成了观察者另外一层含义,古往今来从来不缺这类人,他有一个时髦的名称:间谍</br>
说了这么多,在代码中如何实现呢,让被观察者持有观察者的对象,在被观察者产生动作后,直接调用观察者的方法,就达到了汇报的目的了。

改进被观察者

修改主函数

public static void main(String[] args) throws InterruptedException {
    //创建被观察者
    RealObservable realObservable = new RealObservable();
    realObservable.eat();
    realObservable.sleep();
}

运行打印结果:</br>
我吃饭了。。。</br>
观察到对方动作,开始汇报</br>
老板,有情况。。。目标在吃饭了</br>
汇报完毕</br>
我睡觉了。。。</br>
观察到对方动作,开始汇报</br>
老板,有情况。。。目标在睡觉了</br>
汇报完毕</br>

打印结果正确,同时效率也提高了。那么是不是这样就够了呢,想一想,我们难道只有一个观察者么,
被观察者的行为难道仅仅是“吃饭”,“睡觉”么,我们应该要达到想增加观察者或者删除观察者,一行代码就搞定的目的,这样的代码才方便后期的拓展和维护是吧。

改进被观察者接口

打印结果:</br>
我吃饭了。。。</br>
我是zhangsan观察到对方动作,开始汇报</br>
老板,有情况。。。目标在吃饭了</br>
汇报完毕</br>
我是lisi观察到对方动作,开始汇报</br>
老板,有情况。。。目标在吃饭了</br>
汇报完毕</br>
我是wangwu观察到对方动作,开始汇报</br>
老板,有情况。。。目标在吃饭了</br>
汇报完毕</br>

现在基本上是算完成了,说了这么多,其实JDK中给我们提供了观察者和被观察者(Observer/Observable),当中的方法也和我们所说的差不多,不过看源码可以看出,JDK中的被观察者是使用线程安全的Vector保存观察者对象。

附上使用代码:

打印结果:</br>
我吃饭了</br>
我是wangwu观察到对方动作,开始汇报</br>
老板,有情况。。。目标在吃饭</br>
汇报完毕</br>
我是lisi观察到对方动作,开始汇报</br>
老板,有情况。。。目标在吃饭</br>
汇报完毕</br>
我是zhangsan观察到对方动作,开始汇报</br>
老板,有情况。。。目标在吃饭</br>
汇报完毕</br>

上一篇 下一篇

猜你喜欢

热点阅读