设计模式-观察者模式(行为型)

定义

  • 定义了对象之间的一对多依赖,让多个观察者对象同时监听某个主题对象,当主题对象发生变化时,它的所有依赖者(观察者)都会受到通知并更新。
  • 比如我们的朋友圈,当某人发布一条动态后,只要你在动态下点赞或者评论后,其他好友对该条动态也点赞或评论时,你就可以收到提示,这就是一种观察者模式的体现。

适用场景

  • 关联行为场景,建立一套触发机制。

优点

  • 抽象耦合
  • 支持广播通信

缺点

  • 避免循环依赖
  • 过多的细节,提高时间消耗和程序的复杂度

代码

由于JDK实现了观察者模式的接口,接下来我们来实现一下微博消息提醒这个效果。

首先我们先来进行抽象

我们操作的步骤是

微博 -》 发布动态 -》其他人接收到动态并且点赞

微博(WeiBo

首先我们要明白,我们观察的是朋友圈,而不是某个动态,当朋友圈内状态有所变化时,通知其他人。

/**
 * 微博 被观察对象
 */
public class WeiBo extends Observable {

    private String name;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    /**
     * 发布动态
     */
    public void publishDynamicState(WeiBo weiBo, DynamicState dynamicState) {
        System.out.println(weiBo.getName() +"发布了一条内容为:" + dynamicState.getContent()+"的动态");
        //改变状态 标识当前对象有所改变
        setChanged();
        //通知观察者
        notifyObservers();
    }
}

我们在这里主要用到父类Observable中的

添加观察者(在测试类中会有体现)

/**
 * Adds an observer to the set of observers for this object, provided
 * that it is not the same as some observer already in the set.
 * The order in which notifications will be delivered to multiple
 * observers is not specified. See the class comment.
 *
 * @param   o   an observer to be added.
 * @throws NullPointerException   if the parameter o is null.
 */
public synchronized void addObserver(Observer o) {
    if (o == null)
        throw new NullPointerException();
    if (!obs.contains(o)) {
        obs.addElement(o);
    }
}

通知观察者

/**
 * If this object has changed, as indicated by the
 * <code>hasChanged</code> method, then notify all of its observers
 * and then call the <code>clearChanged</code> method to indicate
 * that this object has no longer changed.
 * <p>
 * Each observer has its <code>update</code> method called with two
 * arguments: this observable object and the <code>arg</code> argument.
 *
 * @param   arg   any object.
 * @see     java.util.Observable#clearChanged()
 * @see     java.util.Observable#hasChanged()
 * @see     java.util.Observer#update(java.util.Observable, java.lang.Object)
 */
public void notifyObservers(Object arg) {
    /*
     * a temporary array buffer, used as a snapshot of the state of
     * current Observers.
     */
    Object[] arrLocal;

    synchronized (this) {
        /* We don't want the Observer doing callbacks into
         * arbitrary code while holding its own Monitor.
         * The code where we extract each Observable from
         * the Vector and store the state of the Observer
         * needs synchronization, but notifying observers
         * does not (should not).  The worst result of any
         * potential race-condition here is that:
         * 1) a newly-added Observer will miss a
         *   notification in progress
         * 2) a recently unregistered Observer will be
         *   wrongly notified when it doesn't care
         */
        if (!changed)
            return;
        arrLocal = obs.toArray();
        clearChanged();
    }

    for (int i = arrLocal.length-1; i>=0; i--)
        ((Observer)arrLocal[i]).update(this, arg);
}

改变状态,允许通知

/**
 * Marks this <tt>Observable</tt> object as having been changed; the
 * <tt>hasChanged</tt> method will now return <tt>true</tt>.
 */
protected synchronized void setChanged() {
    changed = true;
}

以上三个方法是我们经常用到的。

微博动态(DynamicState)

/**
 * 微博动态
 */
public class DynamicState {
    //微博动态内容
    private String content;

    public String getContent() {
        return content;
    }

    public void setContent(String content) {
        this.content = content;
    }

}

微博中的朋友(OtherFriends)

/**
 * 其他朋友 观察者
 */
public class OtherFriends implements Observer {

    //朋友名称
    private String name;

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

    @Override
    public void update(Observable o, Object arg) {
        WeiBo circleOfFriends = (WeiBo) o;
        System.out.println("朋友;" + name + "在"+circleOfFriends.getName()+"的微博,发布的动态:下点赞了。");
    }
}

朋友作为观察者,在点赞后可以得到其他人的点在通知。

UML

observerUML.jpg

我们清晰的看到 观察者和被观察者的依赖关系。

测试

public class ObserverTest {

    public static void main(String[] args) {
        /**
         * 李雷的微博
         */
        WeiBo weiBo = new WeiBo();
        weiBo.setName("Lilei");
        /**
         * 李雷的好友韩梅梅
         */
        OtherFriends otherFriends1 = new OtherFriends("HanMeimei");
        /**
         * 李雷的好友Bob
         */
        OtherFriends otherFriends2 = new OtherFriends("Bob");
        /**
         * 关注了李雷的微博
         */
        weiBo.addObserver(otherFriends1);
        weiBo.addObserver(otherFriends2);
        /**
         * 李雷发布新动态
         */
        DynamicState dynamicState = new DynamicState();
        dynamicState.setContent("How are you?");
        weiBo.publishDynamicState(weiBo,dynamicState);
    }
}

结果

Lilei发布了一条内容为:How are you?的动态
朋友;Bob在Lilei的微博,发布的动态:下点赞了。
朋友;HanMeimei在Lilei的微博,发布的动态:下点赞了。

其他开源

Spring 事件

事件监听接口ApplicationListener 相当于 java.util.Observer

@FunctionalInterface
public interface ApplicationListener<E extends ApplicationEvent> extends EventListener {

   /**
    * Handle an application event.
    * @param event the event to respond to
    */
   void onApplicationEvent(E event);

}

事件发布接口 ApplicationEventPublisherAware 相当于 java.util.Observable

/**
 * Interface to be implemented by any object that wishes to be notified
 * of the ApplicationEventPublisher (typically the ApplicationContext)
 * that it runs in.
 *
 * @author Juergen Hoeller
 * @author Chris Beams
 * @since 1.1.1
 * @see ApplicationContextAware
 */
public interface ApplicationEventPublisherAware extends Aware {

   /**
    * Set the ApplicationEventPublisher that this object runs in.
    * <p>Invoked after population of normal bean properties but before an init
    * callback like InitializingBean's afterPropertiesSet or a custom init-method.
    * Invoked before ApplicationContextAware's setApplicationContext.
    * @param applicationEventPublisher event publisher to be used by this object
    */
   void setApplicationEventPublisher(ApplicationEventPublisher applicationEventPublisher);

}

当然 Spring中的具体实现比较复杂,这里只是给出了接口定义,感兴趣的小伙伴可以自己实现一个事件发布监听就理解其中的道理了。也可以参看一下之前我整合 Spring Cloud Bus中的实现

Spring Cloud系列--Spring Cloud Bus(二)

小结

在观察者模式中,其实最难的就是抽象观察者和被观察者之间的联系,如果捋清楚这些,相信实现各种监听和通知推送都不成问题。

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

推荐阅读更多精彩内容