Android編程設(shè)計(jì)模式之觀察者模式實(shí)例詳解
本文實(shí)例講述了Android編程設(shè)計(jì)模式之觀察者模式。分享給大家供大家參考,具體如下:
一、介紹
觀察者模式是一個(gè)使用率非常高的模式,它最常用的地方是GUI系統(tǒng)、訂閱——發(fā)布系統(tǒng)。因?yàn)檫@個(gè)模式的一個(gè)重要作用就是解耦,將被觀察者和觀察者解耦,使得它們之間的依賴性更小,甚至做到毫無(wú)依賴。以GUI系統(tǒng)來說,應(yīng)用的UI具有易變性,尤其是前期隨著業(yè)務(wù)的改變或者產(chǎn)品的需求修改,應(yīng)用界面也會(huì)經(jīng)常性變化,但是業(yè)務(wù)邏輯基本變化不大,此時(shí),GUI系統(tǒng)需要一套機(jī)制來應(yīng)對(duì)這種情況,使得UI層與具體的業(yè)務(wù)邏輯解耦,觀察者模式此時(shí)就派上用場(chǎng)了。
二、定義
定義對(duì)象間一種一對(duì)多的依賴關(guān)系,使得每當(dāng)一個(gè)對(duì)象改變狀態(tài),則所有依賴于它的對(duì)象都會(huì)得到通知并被自動(dòng)更新。
三、使用場(chǎng)景
關(guān)聯(lián)行為場(chǎng)景,需要注意的是,關(guān)聯(lián)行為是可拆分的,而不是”組合“關(guān)系。
事件多級(jí)觸發(fā)場(chǎng)景。
跨系統(tǒng)的消息交換場(chǎng)景,如消息隊(duì)列、事件總線的處理機(jī)制。
四、觀察者模式的UML類圖
UML類圖:
角色介紹:
Subject:抽象主題,也就是被觀察者(Observable)的角色,抽象主題角色把所有觀察者對(duì)象的引用保存到一個(gè)聚集里,每個(gè)主題都可以有任何數(shù)量的觀察者。抽象主題提供一個(gè)接口,可以增加和刪除觀察者對(duì)象。
ConcreteSubject:具體主題,該角色將有關(guān)狀態(tài)存入具體觀察者對(duì)象,在具體主題的內(nèi)部狀態(tài)發(fā)生改變時(shí),給所有注冊(cè)過的觀察者發(fā)出通知,具體主題角色又叫做具體被觀察者(ConcreteObservable)角色。
Observer:抽象觀察者,該角色是觀察者的抽象類,它定義了一個(gè)更新接口,使得在得到主題的更改通知時(shí)更新自己。
ConcreteObserver:具體的觀察者,該角色實(shí)現(xiàn)抽象觀察者角色所定義的更新接口,以便主題的狀態(tài)發(fā)生改變化時(shí)更新自身的狀態(tài)。
五、簡(jiǎn)單實(shí)現(xiàn)
這里舉一個(gè)追劇的例子,平常為了不錯(cuò)過最新的電視劇我們會(huì)訂閱或關(guān)注這個(gè)電視劇,當(dāng)電視劇更新后會(huì)第一時(shí)間推送給我們,下來就簡(jiǎn)單實(shí)現(xiàn)一下。
抽象觀察者類:
/** * 抽象觀察者類,為所有具體觀察者定義一個(gè)接口,在得到通知時(shí)更新自己 */ public interface Observer { /** * 有更新 * * @param message 消息 */ public void update(String message); }
抽象被觀察者類:
/** * 抽象被觀察者類 */ public interface Observable { /** * 推送消息 * * @param message 內(nèi)容 */ void push(String message); /** * 訂閱 * * @param observer 訂閱者 */ void register(Observer observer); }
具體的觀察者類:
/** * 具體的觀察者類,也就是訂閱者 */ public class User implements Observer { @Override public void update(String message) { System.out.println(name + "," + message + "更新了!"); } // 訂閱者的名字 private String name; public User(String name) { this.name = name; } }
具體的被觀察者類:
/** * 具體的被觀察者類,也就是訂閱的節(jié)目 */ public class Teleplay implements Observable{ private List<Observer> list = new ArrayList<Observer>();//儲(chǔ)存訂閱者 @Override public void push(String message) { for(Observer observer:list){ observer.update(message); } } @Override public void register(Observer observer) { list.add(observer); } }
實(shí)現(xiàn):
public class Client { public static void main(String[] args) { //被觀察者,這里就是用戶訂閱的電視劇 Teleplay teleplay = new Teleplay(); //觀察者,這里就是訂閱用戶 User user1 = new User("小明"); User user2 = new User("小光"); User user3 = new User("小蘭"); //訂閱 teleplay.register(user1); teleplay.register(user2); teleplay.register(user3); //推送新消息 teleplay.push("xxx電視劇"); } }
結(jié)果:
小明,xxx電視劇更新了! 小光,xxx電視劇更新了! 小蘭,xxx電視劇更新了!
由上面的代碼可以看出實(shí)現(xiàn)了一對(duì)多的消息推送,推送消息都是依賴Observer和Observable這些抽象類,而User和Teleplay完全沒有耦合,保證了訂閱系統(tǒng)的靈活性和可擴(kuò)展性。
六、Android源碼中的觀察者模式
1、BaseAdapter
BaseAdapter我相信大家都不陌生,在ListView的適配器中我們都是繼承它。下面來簡(jiǎn)單分析分析。
BaseAdapter 部分代碼:
public abstract class BaseAdapter implements ListAdapter, SpinnerAdapter { //數(shù)據(jù)集觀察者 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); } /** * 當(dāng)數(shù)據(jù)集變化時(shí),通知所有觀察者 */ public void notifyDataSetChanged() { mDataSetObservable.notifyChanged(); } }
看看mDataSetObservable.notifyChanged()
方法:
public class DataSetObservable extends Observable<DataSetObserver> { /** * 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(); } } } }
可以看出在mDataSetObservable.notifyChanged()
中遍歷所有觀察者,并調(diào)用他們的onChanged()
,從而告知觀察者發(fā)生了什么。
那么觀察者怎么來的,那就是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);//注冊(cè)觀察者 ......省略 } }
AdapterDataSetObserver定義在ListView的父類AbsListView中,是一個(gè)數(shù)據(jù)集觀察者,代碼:
class AdapterDataSetObserver extends AdapterView<ListAdapter>.AdapterDataSetObserver { @Override public void onChanged() { super.onChanged(); if (mFastScroller != null) { mFastScroller.onSectionsChanged(); } } @Override public void onInvalidated() { super.onInvalidated(); if (mFastScroller != null) { mFastScroller.onSectionsChanged(); } } }
它由繼承自AbsListView的父類AdapterView的AdapterDataSetObserver
, 代碼如下 :
class AdapterDataSetObserver extends DataSetObserver { private Parcelable mInstanceState = null; // 上文有說道,調(diào)用Adapter的notifyDataSetChanged的時(shí)候會(huì)調(diào)用所有觀察者的onChanged方法,核心實(shí)現(xiàn)就在這里 @Override public void onChanged() { mDataChanged = true; mOldItemCount = mItemCount; // 獲取Adapter中數(shù)據(jù)的數(shù)量 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(); // 重新布局ListView、GridView等AdapterView組件 requestLayout(); } // 代碼省略 public void clearSavedState() { mInstanceState = null; } }
當(dāng)ListView的數(shù)據(jù)發(fā)生變化時(shí),調(diào)用Adapter的notifyDataSetChanged
函數(shù),這個(gè)函數(shù)又會(huì)調(diào)用DataSetObservable
的notifyChanged
函數(shù),這個(gè)函數(shù)會(huì)調(diào)用所有觀察者 (AdapterDataSetObserver) 的onChanged
方法。這就是一個(gè)觀察者模式!
七、總結(jié)
優(yōu)點(diǎn):
觀察者和被觀察者之間是抽象耦合,應(yīng)對(duì)業(yè)務(wù)變化。
增強(qiáng)系統(tǒng)的靈活性和可擴(kuò)展性。
缺點(diǎn):
在應(yīng)用觀察者模式時(shí)需要考慮一下開發(fā)效率和運(yùn)行效率的問題,程序中包括一個(gè)被觀察者、多個(gè)觀察者,開發(fā)、調(diào)試等內(nèi)容會(huì)比較復(fù)雜,而且在Java中消息的通知一般是順序執(zhí)行,那么一個(gè)觀察者卡頓,會(huì)影響整體的執(zhí)行效率,在這種情況下,一般會(huì)采用異步實(shí)現(xiàn)。
更多關(guān)于Android相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《Android開發(fā)入門與進(jìn)階教程》、《Android調(diào)試技巧與常見問題解決方法匯總》、《Android基本組件用法總結(jié)》、《Android視圖View技巧總結(jié)》、《Android布局layout技巧總結(jié)》及《Android控件用法總結(jié)》
希望本文所述對(duì)大家Android程序設(shè)計(jì)有所幫助。
相關(guān)文章
Android之獲取手機(jī)內(nèi)部及sdcard存儲(chǔ)空間的方法
今天小編就為大家分享一篇Android之獲取手機(jī)內(nèi)部及sdcard存儲(chǔ)空間的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-08-08Android開發(fā)之使用150行代碼實(shí)現(xiàn)滑動(dòng)返回效果
本文給大家分享Android開發(fā)之使用150行代碼實(shí)現(xiàn)滑動(dòng)返回效果的代碼,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2019-05-05Android開發(fā)實(shí)現(xiàn)自動(dòng)切換文字TextSwitcher功能示例
這篇文章主要介紹了Android開發(fā)實(shí)現(xiàn)自動(dòng)切換文字TextSwitcher功能,結(jié)合實(shí)例形式詳細(xì)分析了Android使用TextSwitcher實(shí)現(xiàn)文字自動(dòng)切換的原理、實(shí)現(xiàn)方法及相關(guān)操作注意事項(xiàng),需要的朋友可以參考下2019-03-03Android通過滑動(dòng)實(shí)現(xiàn)Activity跳轉(zhuǎn)(手勢(shì)識(shí)別器應(yīng)用)
這篇文章主要為大家詳細(xì)介紹了Android通過滑動(dòng)實(shí)現(xiàn)Activity跳轉(zhuǎn),,講解手勢(shì)識(shí)別器應(yīng)用,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-05-05Android 百度地圖定位實(shí)現(xiàn)仿釘釘簽到打卡功能的完整代碼
這篇文章主要介紹了Android 百度地圖定位實(shí)現(xiàn)仿釘釘簽到打卡功能,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-04-04Android編程使用自定義View實(shí)現(xiàn)水波進(jìn)度效果示例
這篇文章主要介紹了Android編程使用自定義View實(shí)現(xiàn)水波進(jìn)度效果,結(jié)合實(shí)例形式詳細(xì)分析了Android水波動(dòng)畫效果的具體實(shí)現(xiàn)步驟與相關(guān)注意事項(xiàng),需要的朋友可以參考下2017-01-01