Android設(shè)計(jì)模式(十一)-觀察者模式

觀察者模式是一種使用頻率非常高的設(shè)計(jì)模式愕撰,最常用的地方就是訂閱-發(fā)布系統(tǒng)册着。

這個(gè)模式的重要作用就是將觀察者和被觀察者解耦,使他們之間的依賴更小甚至沒(méi)有坷澡。

博客地址,歡迎踩踩

定義

定義對(duì)象一種一對(duì)多的依賴關(guān)系含蓉,使得每當(dāng)一個(gè)對(duì)象改變狀態(tài)频敛,則所有依賴于他的對(duì)象都會(huì)得到通知并被自動(dòng)更新项郊。

使用場(chǎng)景

  • 關(guān)聯(lián)行為場(chǎng)景,這個(gè)關(guān)聯(lián)是可拆分的斟赚。將觀察者和被觀察者封裝在不同的對(duì)象中着降,可以各自獨(dú)立的變化。
  • 當(dāng)一個(gè)對(duì)象改變時(shí)拗军,有其他對(duì)象要進(jìn)行相應(yīng)的變化任洞,但是他并不知道有多少個(gè)對(duì)象需要變化。
  • 跨系統(tǒng)的消息交換長(zhǎng)江发侵,如消息隊(duì)列交掏,時(shí)事件總線等

UML

  • Subject : 抽象被觀察者(Observeable),吧所有觀察者對(duì)象的醫(yī)用保存在一個(gè)集合里刃鳄,每個(gè)主題都可以有任意數(shù)量的觀察者盅弛,抽象被觀察者提供一個(gè)接口,可以增加和刪除觀察者對(duì)象叔锐。
  • ConcreteSubject: 具體的被觀察者挪鹏,將有關(guān)狀態(tài)存入具體的觀察者對(duì)象,在具體的被觀察者內(nèi)部狀態(tài)發(fā)生變化時(shí)愉烙,給所有注冊(cè)的觀察者發(fā)送通知讨盒。
  • Observer : 抽象觀察者,定義了一個(gè)更新接口齿梁,使得在得到被觀察者的通知時(shí)更新自己催植。
  • ConcreteObserver : 具體的觀察者,實(shí)現(xiàn)了抽象觀察者鎖定義的接口勺择,用來(lái)在收到通知時(shí)更新自己。

簡(jiǎn)單實(shí)現(xiàn)

訂閱模式就是個(gè)觀察者模式伦忠,訂閱后省核,被訂閱的有更新就會(huì)提示你。

拿微信公眾號(hào)舉個(gè)例子吧.Java提供的有Observer和Observable類昆码,可以很方便的實(shí)現(xiàn)觀察者模式气忠。

先定義一個(gè)訂閱者,實(shí)現(xiàn)更新方法赋咽。

public class User implements Observer {
    public String name;

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

    @Override
    public void update(Observable o, Object arg) {
        System.out.println("Hi "+name +",公眾號(hào)更新了內(nèi)容:"+arg);
    }
}

定義一個(gè)可觀察者旧噪,有變化時(shí)發(fā)布更新通知。

public class Gamedaily extends Observable {
    public void postNewArticle(String content){
        //內(nèi)容發(fā)生改變
        setChanged();
        //通知所有訂閱者改變的內(nèi)容
        notifyObservers(content);
    }
}

使用

public class Client {
    public static void main(String[] args) {
        Gamedaily gamedaily = new Gamedaily();
        User user1 = new User("user1");
        User user2 = new User("user2");
        User user3 = new User("user3");
        //將觀察者注冊(cè)到可觀察者的通知列表中脓匿。
        gamedaily.addObserver(user1);
        gamedaily.addObserver(user2);
        gamedaily.addObserver(user3);

        gamedaily.postNewArticle("新文章來(lái)了");
    }
}

輸出


當(dāng)公眾號(hào)發(fā)布新文章的時(shí)候淘钟,所有訂閱者都收到的通知,并作出相應(yīng)的改變陪毡。一個(gè)公眾號(hào)對(duì)應(yīng)多個(gè)訂閱者米母,并且完全沒(méi)有耦合勾扭。

Android源碼中的觀察者模式

通常在ListView的內(nèi)容變化時(shí),我們會(huì)調(diào)用notifyDataSetChanged()這個(gè)方法铁瞒,然后ListView里面的數(shù)據(jù)就會(huì)進(jìn)行更新妙色。這個(gè)感覺(jué)就像是觀察者模式。ListView在觀察者內(nèi)容慧耍,內(nèi)容變化發(fā)布通知之后ListView就會(huì)更新數(shù)據(jù)身辨。

看一下這個(gè)方法。

package android.widget;
public abstract class BaseAdapter implements ListAdapter, SpinnerAdapter {
    private final DataSetObservable mDataSetObservable = new DataSetObservable();
    public void registerDataSetObserver(DataSetObserver observer) {
        mDataSetObservable.registerObserver(observer);
    }

    public void unregisterDataSetObserver(DataSetObserver observer) {
        mDataSetObservable.unregisterObserver(observer);
    }
    ......
    public void notifyDataSetChanged() {
        mDataSetObservable.notifyChanged();
    }
    ......
}

這段代碼可以看到這應(yīng)該是一個(gè)觀察者模式芍碧,而且這個(gè)一個(gè)被觀察者煌珊,里面提供了注冊(cè)和注銷觀察者以及通知觀察者的方法。

這些方法是通過(guò)DataSetObservable這個(gè)類調(diào)用的:

package android.database;
public class DataSetObservable extends Observable<DataSetObserver> {   
    public void notifyChanged() {
        synchronized(mObservers) {
            for (int i = mObservers.size() - 1; i >= 0; i--) {
                mObservers.get(i).onChanged();
            }
        }
    }
    ......
}

這個(gè)類繼承自O(shè)bservable<T>师枣,Observable<T>中有一個(gè)protected final ArrayList<T> mObservers = new ArrayList<T>();,
用來(lái)保存注冊(cè)的觀察者怪瓶。mDataSetObservable.registerObserver(observer)mDataSetObservable.unregisterObserver(observer)分別就是增加和刪除。

notifyChanged方法中践美,循環(huán)這個(gè)集合洗贰,調(diào)用每一個(gè)觀察者的onChanged()方法。

那么這些觀察者是什么時(shí)候注冊(cè)的呢陨倡?也就是ListView和Adapter什么時(shí)候成了訂閱關(guān)系敛滋。在ListView的setAdapter()

public class ListView extends AbsListView {
    public void setAdapter(ListAdapter adapter) {
        //如果已經(jīng)有了一個(gè)adapter,注銷這個(gè)adapter之前的觀察者兴革,
        if (mAdapter != null && mDataSetObserver != null) {
            mAdapter.unregisterDataSetObserver(mDataSetObserver);
        }

       ......
       if (mHeaderViewInfos.size() > 0|| mFooterViewInfos.size() > 0) {
            mAdapter = wrapHeaderListAdapterInternal(mHeaderViewInfos, mFooterViewInfos, adapter);
        } else {
            //將新的adapter賦給mAdapter
            mAdapter = adapter;
        }
        ......
        super.setAdapter(adapter);

        if (mAdapter != null) {
            mAreAllItemsSelectable = mAdapter.areAllItemsEnabled();
            //保存之前的數(shù)據(jù)個(gè)數(shù)
            mOldItemCount = mItemCount;
            //獲取新的個(gè)數(shù)
            mItemCount = mAdapter.getCount();
            checkFocus();
            //創(chuàng)建數(shù)據(jù)集觀察者
            mDataSetObserver = new AdapterDataSetObserver();
            //注冊(cè)觀察者
            mAdapter.registerDataSetObserver(mDataSetObserver);
            ...
            }
        } else {
            ...
        }

        requestLayout();
    }
}

AdapterDataSetObserver是ListView的父類AbsListView的內(nèi)部類

package android.widget;
public abstract class AbsListView extends AdapterView<ListAdapter> implements TextWatcher,
        ViewTreeObserver.OnGlobalLayoutListener, Filter.FilterListener,
        ViewTreeObserver.OnTouchModeChangeListener,
        RemoteViewsAdapter.RemoteAdapterConnectionCallback {
       class AdapterDataSetObserver extends AdapterView<ListAdapter>.AdapterDataSetObserver {
        @Override
        public void onChanged() {
            super.onChanged();
            if (mFastScroll != null) {
                mFastScroll.onSectionsChanged();
            }
        }
        ······
    }
}

AdapterDataSetObserver是AdapterView<ListAdapter>.AdapterDataSetObserver的子類绎晃,所以要看super.onChanged()

package android.widget;
public abstract class AdapterView<T extends Adapter> 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();
        }
        ......
    }
}

整理一下:當(dāng)ListView數(shù)據(jù)變化時(shí),調(diào)用Adapter的notifyDataSetChange方法杂曲,這個(gè)方法調(diào)用DataSetObservable的notifyChanged方法庶艾,這個(gè)方法又會(huì)調(diào)用所有觀察者的onChanged方法,onChanged再調(diào)用重新布局View的方法擎勘,完成刷新數(shù)據(jù)的功能咱揍。

總結(jié)

優(yōu)點(diǎn)

  • 解除了觀察者和被觀察者的耦合,而且依賴的都是抽象棚饵,容易應(yīng)對(duì)業(yè)務(wù)變化煤裙,各自的變化都不會(huì)影響另一個(gè)。
  • 增強(qiáng)系統(tǒng)靈活性噪漾、可拓展性硼砰。

缺點(diǎn)

  • Java中的消息默認(rèn)是順序執(zhí)行,如果一個(gè)觀察者卡頓欣硼,會(huì)造成整個(gè)系統(tǒng)效率變低题翰,可以考慮異步。
  • 可能會(huì)引起無(wú)用的操作甚至錯(cuò)誤的操作。
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末遍愿,一起剝皮案震驚了整個(gè)濱河市存淫,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌沼填,老刑警劉巖桅咆,帶你破解...
    沈念sama閱讀 216,591評(píng)論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異坞笙,居然都是意外死亡岩饼,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,448評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門薛夜,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)籍茧,“玉大人,你說(shuō)我怎么就攤上這事梯澜∧耄” “怎么了?”我有些...
    開(kāi)封第一講書人閱讀 162,823評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵晚伙,是天一觀的道長(zhǎng)吮龄。 經(jīng)常有香客問(wèn)我,道長(zhǎng)咆疗,這世上最難降的妖魔是什么漓帚? 我笑而不...
    開(kāi)封第一講書人閱讀 58,204評(píng)論 1 292
  • 正文 為了忘掉前任,我火速辦了婚禮午磁,結(jié)果婚禮上尝抖,老公的妹妹穿的比我還像新娘。我一直安慰自己迅皇,他們只是感情好昧辽,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,228評(píng)論 6 388
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著登颓,像睡著了一般奴迅。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上挺据,一...
    開(kāi)封第一講書人閱讀 51,190評(píng)論 1 299
  • 那天,我揣著相機(jī)與錄音脖隶,去河邊找鬼扁耐。 笑死,一個(gè)胖子當(dāng)著我的面吹牛产阱,可吹牛的內(nèi)容都是我干的婉称。 我是一名探鬼主播,決...
    沈念sama閱讀 40,078評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼王暗!你這毒婦竟也來(lái)了悔据?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書人閱讀 38,923評(píng)論 0 274
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤俗壹,失蹤者是張志新(化名)和其女友劉穎科汗,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體绷雏,經(jīng)...
    沈念sama閱讀 45,334評(píng)論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡头滔,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,550評(píng)論 2 333
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了涎显。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片坤检。...
    茶點(diǎn)故事閱讀 39,727評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖期吓,靈堂內(nèi)的尸體忽然破棺而出早歇,到底是詐尸還是另有隱情,我是刑警寧澤讨勤,帶...
    沈念sama閱讀 35,428評(píng)論 5 343
  • 正文 年R本政府宣布箭跳,位于F島的核電站,受9級(jí)特大地震影響悬襟,放射性物質(zhì)發(fā)生泄漏衅码。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,022評(píng)論 3 326
  • 文/蒙蒙 一脊岳、第九天 我趴在偏房一處隱蔽的房頂上張望逝段。 院中可真熱鬧,春花似錦割捅、人聲如沸奶躯。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 31,672評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)嘹黔。三九已至,卻和暖如春莫瞬,著一層夾襖步出監(jiān)牢的瞬間儡蔓,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 32,826評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工疼邀, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留喂江,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 47,734評(píng)論 2 368
  • 正文 我出身青樓旁振,卻偏偏與公主長(zhǎng)得像获询,于是被迫代替她去往敵國(guó)和親涨岁。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,619評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容