设计模式之观察者模式(observer pattern)

观察者模式主要用于处理对象间的一对多的关系,是一种对象行为模式。该模式的实际应用场景比较容易确认,当一个对象状态发生变化时,所有该对象的关注者均能收到状态变化通知,以进行相应的处理。
本文希望通过简单的介绍和分析,能让读者对观察者模式有一个简单直观的认识和感知,以便在实际开发中根据需要灵活运用。

1. 目的

建立对象间一对多的关联关系,并能使一个对象的变化被所有关联对象感知。

2. 动机

建立一套低耦合的消息触发机制。

3. 优缺点

优点:

  1. 被观察者和观察者之间是抽象耦合的;

  2. 耦合度较低,两者之间的关联仅仅在于消息的通知;

  3. 被观察者无需关心他的观察者;

  4. 支持广播通信;

缺点:

  1. 观察者只知道被观察对象发生了变化,但不知变化的过程和缘由;

  2. 观察者同时也可能是被观察者,消息传递的链路可能会过长,完成所有通知花费时间较多;

  3. 如果观察者和被观察者之间产生循环依赖,或者消息传递链路形成闭环,会导致无限循环;

4. 应用场景

  • 需要在系统中建立一个单项广播的触发机制;

  • 系统中某个对象的行为会影响若干其他对象;

  • 对象之间的关联关系可以在运行时动态的建立与撤销;

  • 对象之间的关联关系呈现出一种树状结构;

5.  原理

下面是GoF介绍的典型的类观察者模式的UML类图:

Subject:

 抽象被观察者,仅提供注册和删除观察者对象的接口声明。

ConcreteSubject:

 具体被观察者对象,该对象中收集了所有需要被通知的观察者,并可以动态的增删集合中的观察者。当其状态发生变化时会通知所有观察者对象。

Observer:

 抽象观察者,为所有观察者定义获得通知的统一接口;

ConcreteObserver:

 观察者对象,其关注对象为Subject,能接受Subject变化时发出的通知并更新自身状态。

6.实现

接下来先将上面的UML类图转换为具体的代码,然后在举一个具体的例子来看一下其应用。

抽象被观察者类:Subject

public interface Subject {public void setState(int state);public int getState();public void attach(Observer obs);public void detach(Observer obs);public void notify(String msg);
}

抽象观察者类:Observer

public interface Observer {public void update(String msg);
}

具体被观察者类:ConcreteSubject

public class ConcreteSubject implements Subject {    private List<Observer> observerList = new ArrayList<Observer>();private int state;

    @Overridepublic void setState(int state) {this.state = state;
        notify("new state: " + state);
    }

    @Overridepublic int getState() {// TODO Auto-generated method stubreturn 0;
    }

    @Overridepublic void attach(Observer obs) {// TODO Auto-generated method stub        observerList.add(obs);
    }

    @Overridepublic void detach(Observer obs) {// TODO Auto-generated method stub        observerList.remove(obs);
    }

    @Overridepublic void notify(String msg) {// TODO Auto-generated method stubfor (Observer obs: observerList) {
            obs.update(msg);
        }
    }
}

具体观察者类:ConcreteObserver

public class ConcreteObserver implements Observer {

    @Overridepublic void update(String msg) {// TODO Auto-generated method stubSystem.out.println("ConcreteObserver receive notify msg: " + msg);
    }

}

演示:

public class Demo {public static void main(String[] args) {
        ConcreteObserver obs = new ConcreteObserver();
        ConcreteSubject sub = new ConcreteSubject();
        sub.attach(obs);
        sub.setState(666);
        sub.notify("just test subject notify function!");
    }
}

结果:

ConcreteObserver receive notify msg: new state: 666ConcreteObserver receive notify msg: just test subject notify function!

7.实例

我们以一个更加实际的例子——商品价格的变动来体会一下观察者模式的用途。

在网上购物的时候,商品一般都有一个价格变动通知,前提是我们关注了该商品。

这里我们稍微变通一下,只有当关注的商品价格下降,且低于用户期望购买价格的时候,才会给用户发送一条商品降价的短信通知。

下面分别定义每个类:

产品抽象类:Product

public interface Product {public void setPrice(int price);public int getPrice();public void follow(User user);public void unfollow(User user);public void notifyLowPrice();
}

用户抽象类:User

public interface User {public boolean isExpectedPrice(int price);public void shortMSG(String msg);
}

商品笔记本电脑:Laptop

public class Laptop implements Product {    private List<User> followList = new ArrayList<User>();private int curPrice;

    @Overridepublic void setPrice(int price) {
        curPrice = price;
        System.out.println("set laptop price: " + price);
        notifyLowPrice();
    }

    @Overridepublic int getPrice() {return curPrice;
    }

    @Overridepublic void follow(User user) {
        followList.add(user);
    }

    @Overridepublic void unfollow(User user) {
        followList.remove(user);
    }

    @Overridepublic void notifyLowPrice() {
        String msg = "" + curPrice;for (User user: followList) {if (user.isExpectedPrice(curPrice)) {
                user.shortMSG(msg);
            }
        }
    }
}

关注笔记本电脑用户类:LaptopBuyer

public class LaptopBuyer implements User {private int expectedPrice;private String userName;public LaptopBuyer(String userName, int expectedPrice) {this.userName = userName;this.expectedPrice = expectedPrice;
    }

    @Overridepublic boolean isExpectedPrice(int curPrice) {// TODO Auto-generated method stubreturn curPrice <= expectedPrice;
    }

    @Overridepublic void shortMSG(String msg) {// TODO Auto-generated method stubSystem.out.println("Your follow product have a low price: " + msg + " TO:" + userName);
    }

}

演示:

public class Demo {public static void main(String[] args) {
        LaptopBuyer Alice = new LaptopBuyer("Alice", 6000);
        LaptopBuyer Jack = new LaptopBuyer("Jack", 6500);
        Laptop laptop = new Laptop();
        laptop.follow(Alice);
        laptop.follow(Jack);
        laptop.setPrice(7000);
        laptop.setPrice(6500);
        laptop.setPrice(6000);
        laptop.unfollow(Jack);
        laptop.setPrice(5999);
        laptop.setPrice(6099);
    }
}

结果:

set laptop price: 7000set laptop price: 6500Your follow product have a low price: 6500 TO:Jack
set laptop price: 6000Your follow product have a low price: 6000 TO:Alice
Your follow product have a low price: 6000 TO:Jack
set laptop price: 5999Your follow product have a low price: 5999 TO:Alice
set laptop price: 6099

上面的这个例子是一个能够很好地解释观察者模式的一个实际用途。

8. 总结

相比较与观察者模式,我们或许有许多获取另外一个对象状态的方式,比如,常见的轮询方式,或者仅仅在需要的时候去查一下对方的状态等,不过观察者模式有其特殊的用途,而且更加灵活。

该模式原理比较简单直接,但是实际使用过程中需要考虑一些细节问题:

  • 何时通知?

  • 有谁触发通知?

  • 观察者是关注状态变化的次数还是最终的状态?

  • 如果消息通知被阻塞,应该怎么办?

  • 是否可以改为异步消息通知?

上面这些都是实际使用时应该考虑的。考虑清楚这些细节才能更灵活的应用该模式解决实际问题。

参考:

GoF《Design Patterns: Elements of Reusable Object-Oriented Software》

https://www.runoob.com/design-pattern/observer-pattern.html

(0)

相关推荐

  • 行为型模式之观察者模式

    在现实世界中,许多对象并不是独立存在的,其中一个对象的行为发生改变可能会导致一个或者多个其他对象的行为也发生改变.例如,某种商品的物价上涨时会导致部分商家高兴,而消费者伤心. 在软件世界也是这样,例如 ...

  • 通俗易懂系列 | 设计模式(七):观察者模式

    介绍# 观察者模式是行为设计模式之一.当您对对象的状态感兴趣并希望在有任何更改时收到通知时,观察者设计模式非常有用.在观察者模式中,监视另一个对象状态的对象称为Observer,正在被监视的对象称为S ...

  • 软件设计模式修炼 -- 观察者模式

    观察者模式是一种经常使用的设计模式,在软件系统中对象并不是孤立存在的,一个对象行为的改变可能会导致其他与之存在依赖关系的对象行为发生改变,观察者模式用于描述对象之间的依赖关系. 模式动机 很多情况下, ...

  • 谈谈Java常用类库中的设计模式 - Part Ⅲ

    概述 本系列上一篇:适配器.模版方法.装饰器 本文介绍的设计模式: 策略 观察者 代理 相关缩写:EJ - Effective Java Here We Go 策略 (Stragety) 定义:定义算 ...

  • JAVA设计模式之观察者模式 - Observer

    有趣的事情发生时,可千万别错过了!有一个模式可以帮你的对象知悉现况,不会错过该对象感兴趣的事.对象甚至在运行时可决定是否要继续被通知.有了观察者,你将会消息灵通. 介绍 观察者模式的定义: 在对象之间 ...

  • 【2/25】在Game上应用观察者模式(Observer Pattern)

    这是<小游戏从0到1设计模式重构>系列内容第2篇,所有源码及资料在"程序员LIYI"公号回复"小游戏从0到1"获取. 作者使用过的最简洁的观察者模式 ...

  • 设计模式之单例模式(Singleton Pattern)

    一.定义 一个类只有一个实例,且该类能自行创建这个实例的一种模式. 二.单例模式举例 例如,Windows 中只能打开一个任务管理器,这样可以避免因打开多个任务管理器窗口而造成内存资源的浪费,或出现各 ...

  • PHP设计模式之观察者模式

    PHP设计模式之观察者模式 观察者,貌似在很多科幻作品中都会有这个角色的出现.比如我很喜欢的一部美剧<危机边缘>,在这个剧集中,观察者不停的穿越时空记录着各种各样的人或事.但是,设计模式中 ...

  • [PHP小课堂]PHP设计模式之观察者模式

    [PHP小课堂]PHP设计模式之观察者模式 关注公众号:[硬核项目经理]获取最新文章 添加微信/QQ好友:[DarkMatterZyCoder/149844827]免费得PHP.项目管理学习资料

  • 【设计模式】单例模式(Singleton Pattern)

    懒汉式 public class Singleton { private static Singleton instance; private Singleton() {}; public stati ...

  • 【转】C#设计模式-单例模式(Singleton Pattern)

    目录 介绍 第一个版本 --不是线程安全的 第二个版本 -- 简单的线程安全 第三个版本 - 使用双重检查锁定尝试线程安全 第四个版本 - 不太懒,不使用锁且线程安全 第五版 - 完全懒惰的实例化 第 ...

  • 设计模式(20) 观察者模式

    观察者模式是一种平时接触较多的模式.它主要用于一对多的通知发布机制,当一个对象发生改变时自动通知其他对象,其他对象便做出相应的反应,同时保证了被观察对象与观察对象之间没有直接的依赖. GOF对观察者模 ...