观察者模式 观察者模式

一、简介

观察者模式(又被称为发布-订阅(Publish/Subscribe)模式,属于行为型模式的一种,它定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象。这个主题对象在状态变化时,会通知所有的观察者对象,使他们能够自动更新自己。该模式一个重要作用就是解耦,将被观察者和观察者进行解耦,使他们之间的依赖性更小

二、使用场景

  • 关联行为场景,需要注意的是关联行为是可拆分的而不是“组合”关系
  • 事件多级触发场景
  • 跨系统的消息交换场景,如消息队列、事件总线的处理机制

三、简单实现

这里我们以微信公众号的订阅为例。公众号当其更新内容时就会推送给订阅了该公众号的读者。

    //被观察者
    public class Wechat extends Observable{
        public void postNewPublication(String content){
            setChanged();
            notifyObservers(content);
        }
    }

    //观察者
    public class Reader implements Observer{
       public String name ;

        public Reader(String name) {
            this.name = name;
        }

        @Override
        public void update(Observable o, Object arg) {
            Log.i(TAG, "update: wechat is update content is :"+arg);
        }
    }
    
    public void test(){
        Wechat wechat=new Wechat();
        
        Reader reader1=new Reader("reader1");
        Reader reader2=new Reader("reader2");
        Reader reader3=new Reader("reader3");
        
        wechat.addObserver(reader1);
        wechat.addObserver(reader2);
        wechat.addObserver(reader3);
        
        wechat.postNewPublication("up up up");
    }

这里需要注意的是Observer和Observable是JDK内置的类,表示观察者和被观察者。

四、观察者模式在Android中应用

ListView和RecycleView 的notifyDataSetChanged

当我们在使用ListView或RecycleView时如果数据发生变化我们会调用Adapter的notifyDataSetChanged()方法,如下所示

 public void notifyDataSetChanged() {
        mDataSetObservable.notifyChanged();
    }

在方法内部调用了mDataSetObservable.notifyChanged,这里的mDataSetObservable是一个DataSetObservable实例

    private final DataSetObservable mDataSetObservable = new DataSetObservable();

而DataSetObservable继承自Observable,我们看下DataSetObservable的notifyChanged方法

public class DataSetObservable extends Observable<DataSetObserver> {

public void notifyChanged() {
        synchronized(mObservers) {
            for (int i = mObservers.size() - 1; i >= 0; i--) {
                mObservers.get(i).onChanged();
            }
        }
    }
 }

可以看到调用了DataSetObserver的onChanged方法,DataSetObserver是一个抽类这里mObservers.get(i)获得的是其子类AdapterDataSetObserver.

class AdapterDataSetObserver extends DataSetObserver {
        private Parcelable mInstanceState = null;
        @Override
        public void onChanged() {
            mDataChanged = true;
            mOldItemCount = mItemCount;
            mItemCount = getAdapter().getCount();
            if (AdapterView.this.getAdapter().hasStableIds() && mInstanceState != null
                    && mOldItemCount == 0 && mItemCount > 0) {
                AdapterView.this.onRestoreInstanceState(mInstanceState);
                mInstanceState = null;
            } else {
                rememberSyncState();
            }
            checkFocus();
            //重新布局
            requestLayout();
        }

        ...

        public void clearSavedState() {
            mInstanceState = null;
        }
 }

可以看到在AdapterDataSetObserver的onChanged方法中调用了requestLayout来进行重新布局。

BroadcastReceiver

在Android中广播也是基于观察者模式的

五、小结

观察者模式优点:

  • 解耦观察者与被观察者,应对业务变化
  • 增强系统灵活性、可扩展性

缺点:

  • 在使用时要考虑开发效率和运行效率,程序中包括一个被观察者、多个观察者、开发调试等内容会比较复杂,且Java中消息通知默认是顺序执行,如果一个观察者卡顿,那么会影响整体执行效率,在这种情况下一般考虑使用异步的方式。