详解Android观察者模式的使用与优劣
导读
一、简介
观察者模式(又被称为发布-订阅(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中消息通知默认是顺序执行,如果一个观察者卡顿,那么会影响整体执行效率,在这种情况下一般考虑使用异步的方式。