android 网卡启动 观察者模式,android适配器中的观察者模式

1. 模式介绍

模式的定义

定义对象间一种一对多的依赖关系,使得每当一个对象改变状态,则所有依赖于它的对象都会得到通知并被自动更新。

模式的使用场景

关联行为场景。需要注意的是,关联行为是可拆分的,而不是“组合”关系;

事件多级触发场景;

跨系统的消息交换场景,如消息队列、事件总线的处理机制。

2. UML类图

782732b11c17f9fca477803d9fd83929.png

角色介绍

抽象主题 (Subject) 角色  抽象主题角色把所有观察者对象的引用保存在一个聚集(比如ArrayList对象)里,每个主题都可以有任意数量的观察者。抽象主题提供一个接口,可以增加和删除观察者对象,抽象主题角色又叫做抽象被观察者(Observable)角色。

具体主题 (ConcreteSubject) 角色  将有关状态存入具体观察者对象;在具体主题的内部状态改变时,给所有登记过的观察者发出通知。具体主题角色又叫做具体被观察者(Concrete Observable)角色。

抽象观察者 (Observer) 角色  为所有的具体观察者定义一个接口,在得到主题的通知时更新自己,这个接口叫做更新接口。

具体观察者 (ConcreteObserver) 角色  存储与主题的状态自恰的状态。具体观察者角色实现抽象观察者角色所要求的更新接口,以便使本身的状态与主题的状态 像协调。如果需要,具体观察者角色可以保持一个指向具体主题对象的引用。

三.

ListView中观察者模式的使用—继承AdapterView,组合ListAdapter

private void initView() {

mListView = (ListView) findViewById(R.id.list_view);

findViewById(R.id.update_tv).setOnClickListener(this);

mDataList = new ArrayList();

String title = null;

for (int i = 0; i < 20; i++) {

title = new String("title_" + i);

mDataList.add(title);

}

mAdapter = new DataAdapter(this, mDataList);

mListView.setAdapter(mAdapter);

mAdapter.notifyDataSetChanged();

}

@Override

public void onClick(View v) {

if (v.getId() == R.id.update_tv) {

int len = mDataList.size();

int index = new Random().nextInt(len);

mDataList.set(index, "updata_" + index);

mAdapter.notifyDataSetChanged();

}

}

那么我们需要知道mAdapter.notifyDataSetChanged();具体的实现

public abstract class BaseAdapter implements ListAdapter, SpinnerAdapter {

private final DataSetObservable mDataSetObservable = new DataSetObservable();

public boolean hasStableIds() {

return false;

}

public void registerDataSetObserver(DataSetObserver observer) {

mDataSetObservable.registerObserver(observer);

}

public void unregisterDataSetObserver(DataSetObserver observer) {

mDataSetObservable.unregisterObserver(observer);

}

/**

* Notifies the attached observers that the underlying data has been changed

* and any View reflecting the data set should refresh itself.

*/

public void notifyDataSetChanged() {

mDataSetObservable.notifyChanged();

}

。。。

}

点击mDataSetObservable.notifyChanged();

public class DataSetObservable extends Observable {

/**

* Invokes {@link DataSetObserver#onChanged} on each observer.

* Called when the contents of the data set have changed.  The recipient

* will obtain the new contents the next time it queries the data set.

*/

public void notifyChanged() {

synchronized(mObservers) {

// since onChanged() is implemented by the app, it could do anything, including

// removing itself from {@link mObservers} - and that could cause problems if

// an iterator is used on the ArrayList {@link mObservers}.

// to avoid such problems, just march thru the list in the reverse order.

for (int i = mObservers.size() - 1; i >= 0; i--) {

mObservers.get(i).onChanged();

}

}

}

。。。

}

其被观察者(事件源)Observable基类的定义为

/**

* Provides methods for registering or unregistering arbitrary observers in an {@link ArrayList}.

*

* This abstract class is intended to be subclassed and specialized to maintain

* a registry of observers of specific types and dispatch notifications to them.

*

* @param T The observer type.

*/

public abstract class Observable {

/**

* The list of observers.  An observer can be in the list at most

* once and will never be null.

*/

protected final ArrayList mObservers = new ArrayList();

/**

* Adds an observer to the list. The observer cannot be null and it must not already

* be registered.

* @param observer the observer to register

* @throws IllegalArgumentException the observer is null

* @throws IllegalStateException the observer is already registered

*/

public void registerObserver(T observer) {

if (observer == null) {

throw new IllegalArgumentException("The observer is null.");

}

synchronized(mObservers) {

if (mObservers.contains(observer)) {

throw new IllegalStateException("Observer " + observer + " is already registered.");

}

mObservers.add(observer);

}

}

}

到此我们已经知道mAdapter.notifyDataSetChanged();最终是Observable遍历注册的观察者对象集合,执行DataSetObserver.onChanged()。

那么我们就要查看源码看下该观察者被注册到事件源的

直接查看ListView.setAdapter()方法的源码实现

@Override

public void setAdapter(ListAdapter adapter) {

if (mAdapter != null && mDataSetObserver != null) {

mAdapter.unregisterDataSetObserver(mDataSetObserver);

}

resetList();

mRecycler.clear();

if (mHeaderViewInfos.size() > 0|| mFooterViewInfos.size() > 0) {

mAdapter = new HeaderViewListAdapter(mHeaderViewInfos, mFooterViewInfos, adapter);

} else {

mAdapter = adapter;

}

mOldSelectedPosition = INVALID_POSITION;

mOldSelectedRowId = INVALID_ROW_ID;

// AbsListView#setAdapter will update choice mode states.

super.setAdapter(adapter);

if (mAdapter != null) {

mAreAllItemsSelectable = mAdapter.areAllItemsEnabled();

mOldItemCount = mItemCount;

mItemCount = mAdapter.getCount();

checkFocus();

mDataSetObserver = new AdapterDataSetObserver();

mAdapter.registerDataSetObserver(mDataSetObserver);

}

。。。

requestLayout();

}

可以看到在mAdapter.registerDataSetObserver(mDataSetObserver);中已经将观察者对象注册到事件源对象中了。其中mDataSetObserver对象是声明在父类AbsListView中AdapterDataSetObserver mDataSetObserver;而AdapterDataSetObserver的定义是在AdapterView的内部类中。

public abstract class AdapterView extends ViewGroup {

class AdapterDataSetObserver extends DataSetObserver {

private Parcelable mInstanceState = null;

@Override

public void onChanged() {

mDataChanged = true;

mOldItemCount = mItemCount;

mItemCount = getAdapter().getCount();

// Detect the case where a cursor that was previously invalidated has

// been repopulated with new data.

if (AdapterView.this.getAdapter().hasStableIds() && mInstanceState != null

&& mOldItemCount == 0 && mItemCount > 0) {

AdapterView.this.onRestoreInstanceState(mInstanceState);

mInstanceState = null;

} else {

rememberSyncState();

}

checkFocus();

requestLayout();

}

。。。

}

}

ListView中观察者模式的使用—数据变化通知界面更新的流程

1)ListView继承自AdapterView(其中包含内部类AdapterDataSetObserver)

2)当ListView调用setAdapter(ListAdapteradapter)时,将生成观察者实例(ListView充当了观察者角色),并注册到mAdapter中(ListAdapter充当了被观察者角色)。

mDataSetObserver= new AdapterDataSetObserver();

mAdapter.registerDataSetObserver(mDataSetObserver);

3)Adapter中数据变化时,将调用mAdapter.notifyDataSetChanged(),实际调用的是mDataObservable的notifyChanged(),其内部将执行每一个observer的onChanged(),也就达到了更新界面的效果。

观察者模式最佳实践

项目实践中典型的观察者模式框架:EventBus、RxJava、KVC/KVO(iOS)

小结

1.为什么引入设计模式?

使用设计模式有助于软件适应变化,增强可维护性、可复用性。

设计模式遵循的六大设计原则:SOLID+LawofDemeter。“高内聚、低耦合”

2.观察者模式主要作用就是解耦。

将观察者和被观察者进行隔离,只依赖于Observer和Observable抽象。

3.Android中的源代码,有助于我们理解观察者模式的编码实现方式,理解Listview界面更新的背后逻辑。

4.观察者模式最佳实践:EventBus、RxJava、KVC/KVO

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值