观察者设计模式

1. 定义?


当一个对象的状态发生改变时,与他相关联的部分对象的状态同时也会发生改变。

比如订阅公众号:

  • 比如我订阅了鸿阳的微信公众号,只要他每更新一篇文章都会及时的通知我 [观察者设计模式];
    对应上边定义就是:一个对象的状态发生改变时,就是鸿阳公众号有文章更新,由于我订阅了该公众号,所以说我的部分状态也会发生改变,比如我会去看该公众号更新的文章;

EventBus:和观察者设计模式没有半毛钱关系

2. 角色划分?


被观察者(Observable):公众号;
具体的被观察者(Concreate Observable):鸿阳公众号;

观察者(Observer):微信用户;
具体的观察者(Concreate Observer):我,Novate

3. 示例代码 - 订阅公众号?


写一个事例代码
被观察者:WXPublicObservable(公众号);
具体的被观察者:WXAdvanceObservable(鸿阳的公众号);

观察者:IWXUser(微信用户);
具体的观察者:WXUser(Novate、WangZiWen);

4. 示例代码如下


1>:WXPublicObservable,被观察者 - 微信公众号
/**
 * Email: 2185134304@qq.com
 * Created by Novate 2018/5/27 15:03
 * Version 1.0
 * Params:
 * Description:    微信公众号 - 多个人去订阅的公众号
*/

public class WXPublicObservable {

    // 所有订阅用户的集合
    private List<IWXUser> mWXUsers ;
    public WXPublicObservable(){
        mWXUsers = new ArrayList<>() ;
    }

    /**
     * 订阅
     */
    public void register(IWXUser wxUser){
        mWXUsers.add(wxUser) ;
    }

    /**
     * 取消订阅
     */
    public void unregister(IWXUser wxUser){
        mWXUsers.remove(wxUser) ;
    }

    /**
     * 文章更新
     */
    public void update(String article){
        // 推送所有更新的文章
        for (IWXUser wXUser : mWXUsers) {
            wXUser.push(article);
        }
    }
}

2>:WXAdvanceObservable,具体的被观察者 - 鸿阳的微信公众号
/**
 * Email: 2185134304@qq.com
 * Created by Novate 2018/5/27 15:12
 * Version 1.0
 * Params:
 * Description:    具体的被观察者 - 鸿阳的微信公众号
*/

public class WXAdvanceObservable extends WXPublicObservable{

    private String article ;

    public String getArticle() {
        return article;
    }

    public void setArticle(String article) {
        this.article = article;

        // 通知更新,推送给微信用户
        update(article);
    }
}

3>:IWXUser,接口,微信用户;
/**
 * Email: 2185134304@qq.com
 * Created by Novate 2018/5/27 15:06
 * Version 1.0
 * Params:
 * Description:    微信用户 - 订阅该微信公众号
*/

public interface IWXUser {

    /**
     * 读文章
     */
    void push(String article) ;
}

4>:WXUser,具体的观察者,具体的用户,订阅鸿阳微信公众号的用户
/**
 * Email: 2185134304@qq.com
 * Created by Novate 2018/5/27 15:14
 * Version 1.0
 * Params:
 * Description:    具体的用户 - 订阅鸿阳的微信公众号
*/

public class WXUser implements IWXUser{

    private String name ;

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

    @Override
    public void push(String article) {
        System.out.println(name+"收到了一篇文章:"+article);
    }
}

5>:创建具体的被观察者和具体的观察者对象,测试代码如下:
/**
 * Email: 2185134304@qq.com
 * Created by Novate 2018/5/27 15:16
 * Version 1.0
 * Params:
 * Description:
*/

public class Client {
    public static void main(String[] args){

        // 具体的被观察者 - 微信公众号 - 鸿阳的公众号
        WXAdvanceObservable wxAdvanceObservable = new WXAdvanceObservable() ;

        // 具体的观察者 - 微信公众号 - Novate
        WXUser novate = new WXUser("novate") ;
        WXUser wangziwen = new WXUser("wangziwen") ;

        // 微信公众号 - 用户订阅公众号
        wxAdvanceObservable.register(novate);
        wxAdvanceObservable.register(wangziwen);

        // 微信公众号 - 推送文章
        wxAdvanceObservable.setArticle("《观察者设计模式 - 定义及事例代码》");

        //  微信公众号 - 用户取消订阅公众号
        wxAdvanceObservable.unregister(wangziwen);
    }
}

运行结果打印如下

image

5. 源码中观察者设计模式的使用场景


1>:RxJava源码;
2>:ListView的 Adapter的setDataChange的方法;

6. ListView部分源码分析


1>:ListView中的setAdapter()方法
@Override
    public void setAdapter(ListAdapter adapter) {
        // 防止多次调用setAdapter,而不去调用notifyDataSetChanged
        if (mAdapter != null && mDataSetObserver != null) {
            mAdapter.unregisterDataSetObserver(mDataSetObserver);
        }
            // 给adapter注册一个 mDataSetObserver
            mAdapter.registerDataSetObserver(mDataSetObserver);
            // 
            requestLayout();
    }

2>:只要调用了 adapter.notifyDataSetChanged()方法,就会执行下边代码:
A:BaseAdapter中的notifyDataSetChanged():
 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();
    }

B:然后调用
public void notifyChanged() {
        synchronized(mObservers) {

            for (int i = mObservers.size() - 1; i >= 0; i--) {
                // 只要一更新,就会调用onChanged(),
                // 所以其实是用 onChanged()方法把 ListView与adapter进行关联
                mObservers.get(i).onChanged();
            }
        }
    }

3>:这个时候会来到AdapterView的onChanged()方法,来更新ListView;
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();
            // 重新执行onMeasure()、onLayout()、onDraw()这几个方法
            requestLayout();
        }

        @Override
        public void onInvalidated() {
            mDataChanged = true;
            checkFocus();
            requestLayout();
        }

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

7. ListView观察者设计模式图解


image
由以上分析ListView观察者设计模式图解可知:

1>:ListView与adapter二者其实关联不太大,ListView只是调用了setAdapter()方法,那么adapter如果数据改变如何通知ListView刷新界面,比如adapter少了一条数据,就需要ListView少显示一条数据;
2>:其实在ListView调用setAdapter()时候,会给它的adapter中注册一群观察者,也就是说ListView中有 Observer,adapter中有一群Observable,也就是说有多个Observable,把ListView中的Observer注册到adapter中的Observable,也就是说把ListView的对象注册到adapter中的Observable中;
3>:只要调用了 notifySetDataChanged(),这个时候adapter中所有的 Observable会进行for循环来调用 Observer中的onChanged()方法;
4>:然后在AdapterView中,调用onChanged()方法,然后再调用 requestLayout()方法,重新执行onMeasure()、onLayout()、onDraw()方法;

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,843评论 6 502
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,538评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,187评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,264评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,289评论 6 390
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,231评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,116评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,945评论 0 275
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,367评论 1 313
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,581评论 2 333
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,754评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,458评论 5 344
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,068评论 3 327
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,692评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,842评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,797评论 2 369
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,654评论 2 354

推荐阅读更多精彩内容