设计模式学习(二)——观察者模式

一.需求

时间过的真快,小王的便利店已经开了大半年了,生意很是红火。

一天,小王远在乡下的表哥来城里办事,顺便来看小王。表哥在乡下开了个养鸡场,都是散养的柴鸡,每过一段时间,表哥就会来城里送一批柴鸡蛋。

小王突发奇想:能不能在自己的便利店也卖柴鸡蛋呢?这个想法一说,表哥当即就赞同了。几天后,就送来了第一批柴鸡蛋。

小区张大妈经常给孙女做糕点,看到便利店卖柴鸡蛋,就买了一些;便利店隔壁是赵阿姨开的一家餐馆,也买了一些回去。吃过小王便利店里柴鸡蛋的人都说不错,渐渐的,柴鸡蛋开始供不应求了。不少顾客来买东西都要问还有没有柴鸡蛋卖,甚至有一些顾客每天都来一次专门问柴鸡蛋到货了没有。

小王感到很是不安,这么多人来买鸡蛋,但是鸡蛋根本不够卖,害的顾客空跑,有没有什么好办法呢?

二.初步尝试

经过一番仔细分析,小王觉得,眼下最需要解决的问题是让大家知道店里还有没有鸡蛋卖,不能让大家空跑一趟。

小王想到的办法是:在店门口贴一张通知栏,在布告栏上写明当前店里有没有柴鸡蛋卖,并及时更新。每个顾客每天从小区门口路过,如果自己想买柴鸡蛋,并且布告栏上写着有鸡蛋,则可以进店购买。

在这个过程中,有两个对象:通知栏,顾客。通知栏有一个状态来实时展示当前是否有柴鸡蛋,顾客每天看通知栏,决定是否进店购买。如果用程序实现就是下面这样:

// 布告栏
class Notice {
    private boolean hasEgg;

    public boolean hasEgg() {
        return hasEgg;
    }

    public void setHasEgg(boolean hasEgg) {
        this.hasEgg = hasEgg;
    }
}

// 顾客
class Customer {

    private String name;

    private boolean needEgg;

    private Notice notice;

    public Customer(String name, boolean needEgg, Notice notice) {
        this.name = name;
        this.needEgg = needEgg;
        this.notice = notice;
    }

    public void bugEgg() {
        if (needEgg && notice.hasEgg()) {
            System.out.println(this.name + "进店购买鸡蛋");
        } else {
            System.out.println(this.name + "不进店购买鸡蛋");
        }
    }

    public String getName() {
        return name;
    }

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

    public boolean isNeedEgg() {
        return needEgg;
    }

    public void setNeedEgg(boolean needEgg) {
        this.needEgg = needEgg;
    }
}

// 客户端
public class Main {

    public static void main(String[] args) {
        Notice notice = new Notice();
        notice.setHasEgg(true); // 鸡蛋到货了

        Customer motherZhang = new Customer("张大妈", true, notice);
        Customer auntZhao = new Customer("赵阿姨", true, notice);
        Customer uncleLi = new Customer("李叔叔", false, notice);

        motherZhang.bugEgg();
        auntZhao.bugEgg();
        uncleLi.bugEgg();
    }
}

运行一下试试:

张大妈进店购买鸡蛋
赵阿姨进店购买鸡蛋
李叔叔不进店购买鸡蛋

看起来一切正常!所有的顾客不用进店就可以知道有没有柴鸡蛋卖,大大方便了顾客。

三.更好的方案

上面的方案可以正常工作,实现通知的目的。但是仍然有几个问题:
1.通知不够及时,依赖于顾客自己主动去看通知。如果鸡蛋到货时,自己刚好在家里没有看到,就会错过买鸡蛋;
2.一些顾客不关心柴鸡蛋的通知。尽管有很多顾客喜欢小王家的柴鸡蛋,但是也有不少顾客其实并不关心,门口的布告栏对他们来说毫无价值。

第二个问题倒是无关紧要,第一个问题的确是存在的,需要进一步改进。

为了进一步方便大家来买柴鸡蛋,小王想到了一个更好的办法:对于那些经常买鸡蛋的顾客,留下他们的手机号码,一旦有鸡蛋可卖,就给他们群发短信,通知他们可以来买鸡蛋了。

这样一来,就解决了通知不及时的问题,而且第二个问题同时也解决了:短信只会发给那些想买鸡蛋的人,不想买鸡蛋的人不会收到。之前想买鸡蛋但是后来不想买了,可以到小王这里说一下,下次就不会给他发短息了;之前不想买鸡蛋的人也可以随时到小王这里留下电话,下次就可以接到通知了。

新的方案如下:

// 鸡蛋管理
class EggAdmin {
    private List<Customer> customerList;

    private boolean hasEgg;

    public EggAdmin() {
        customerList = new ArrayList<Customer>();
    }

    public void addCustomer(Customer customer) {
        System.out.println(customer.getName() + "订阅鸡蛋到货消息");
        customerList.add(customer);
    }

    public void removeCustomer(Customer customer) {
        System.out.println(customer.getName() + "退订鸡蛋到货消息");
        customerList.remove(customer);
    }

    public void notice() {
        if (hasEgg) { // 一旦鸡蛋到货就通知所有想买鸡蛋的人来买鸡蛋
            for (Customer customer : customerList) {
                customer.bugEgg();
            }
        }
    }

    public boolean hasEgg() {
        return hasEgg;
    }

    public void setHasEgg(boolean hasEgg) {
        this.hasEgg = hasEgg;
    }
}
// 顾客
class Customer {

    private String name;

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

    public void bugEgg() {
        System.out.println(this.name + "进店购买鸡蛋");
    }

    public String getName() {
        return name;
    }

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

    @Override
    public boolean equals(Object obj) {
        if (obj == null) {
            return false;
        }
        if ( ! (obj instanceof  Customer)) {
            return false;
        }
        Customer newCustomer = (Customer) obj;
        return this.name.equals(newCustomer.getName());
    }
}

// 客户端
public class Main {

    public static void main(String[] args) {

        Customer motherZhang = new Customer("张大妈");
        Customer auntZhao = new Customer("赵阿姨");
        Customer uncleLi = new Customer("李叔叔");

        EggAdmin eggAdmin = new EggAdmin();
        eggAdmin.addCustomer(motherZhang);
        eggAdmin.addCustomer(auntZhao);
        eggAdmin.setHasEgg(true);
        eggAdmin.notice();

        eggAdmin.addCustomer(uncleLi);
        eggAdmin.removeCustomer(auntZhao);
        eggAdmin.notice();

    }
}

运行一下:

张大妈订阅鸡蛋到货消息
赵阿姨订阅鸡蛋到货消息
张大妈进店购买鸡蛋
赵阿姨进店购买鸡蛋
李叔叔订阅鸡蛋到货消息
赵阿姨退订鸡蛋到货消息
张大妈进店购买鸡蛋
李叔叔进店购买鸡蛋

感觉还不错,这下想买鸡蛋的顾客就可以及时收到提醒,来店里买鸡蛋了。

四.模式总结

我们在上面的新方案中用到了观察者模式。但是我们只是简单的应用,真实的观察者模式要更为灵活,定义了抽象的主题、具体的主题,抽象的观察者、具体的观察者,采用面向接口编程。

使用场景

当你需要维护一个一对多的关系时,其中一个主题的状态发生变化,其他几个依赖者都需要接收到通知并自动更新。

一个典型的使用场景是:在交互界面,我们可以对每个按钮添加Listener,当按钮被点击时,所有Listener都会得到通知。

类图
观察者模式类图

在观察者模式中,有主题和观察者两种角色,主题可以有多种实现,观察者亦可以有多种实现,每个具体主题中维护了观察者列表,以便通知所有观察者。每个观察者中也维护了需要观察的主题,以便将来取消观察。

在主题中通常还会有一个subjectState属性用来表示状态,这可以更灵活的控制通知发生的时机,例如在我们上面的例子中,有一个hasEgg字段用来表示是否有鸡蛋,只有当有鸡蛋时才会真正发出通知。

优点

1.观察者模式定义了一种稳定的消息传递机制,可以优雅的在主题与观察者之间发布更新;
2.主题和观察者之间是松耦合的,彼此间不知道对方的实现细节,且基于接口编程,当有新的观察者出现时,主题类无需修改,易于扩展;

缺点

如果一个主题的观察者较多,则观察者全部接到通知会花费一定的时间,不同的观察者接收到通知的时间也会有先后,由此可能会导致一些问题。

参考资料:
《Head First Java》

本文已迁移至我的博客:http://ipenge.com/631.html

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

推荐阅读更多精彩内容

  • 早早预定电影票,文艺一回,共赴《大鱼海棠》十二年之约。 走出电影院,心情还算平静。平心而论,大鱼呈现的一切,满足了...
    留逝阅读 452评论 0 1
  • 前些日被前半生这部电视剧刷屏,近今天又被战狼2给刷屏,看着各种主旋律,突然想,我们的生活究竟为了什么? 前半生这部...
    魔王_Archenemy阅读 293评论 0 0
  • W先生和J小姐在租的房子里过着他们从来没想过的生活,躺在W先生只在儿时看到过双人床上,这一切都那么不真实。不过两...
    Celia欢阅读 341评论 0 0