设计模式之订阅/发布模式(subscribe/publish)

一、什么是订阅发布模式

订阅发布模式定义了一种 一对多 的依赖关系,一指发布者,比如某个主题对象,多指订阅者,依赖是订阅者对发布者的依赖;多个订阅者同时监听一个主题对象。当发布者即主题对象的状态发生变化时,会将该变化通知给订阅者,订阅者据此更新自己的状态。是设计模式中用的比较多的一种。

二、为什么使用订阅发布模式

订阅发布模式旨在降低系统不同模块之间的耦合度。一个复杂的系统一般要划分为好多不同的类,这些对象之间并不独立,存在各种协作关系,这就需要维护对象之间的一致性,这给系统的维护、重用和扩展带来不便,严重限制了系统的灵活性。这样就可以把主动变化因素抽象出来形成发布者,而其他依赖者抽象为订阅者,使他们可以独立的被复用和维护。双发都只依赖与抽象,而不是依赖于具体实现。比如发布者新增一个订阅者时只需要发布者注册一下,订阅者新增依赖时,只需要将自己注册到新的发布者即可。

三、一个例子:

有一个天气预报系统,负责通知明天的天气情况。有几个角色订阅了该系统,他们根据明天的天气情况安排明天的事务。这几个角色分别为农夫、建筑工人、程序员。

下雨天
农夫喜出望外:又可以好好歇一天了,让庄稼庄稼好好成长吧;
建筑工人欣喜万分:完美,睡到天昏地暗;
程序员:继续上班,关我毛事;

日晴万里
农夫感慨:即将迎接充实的一天,收获满满;
建筑工人牢骚:这么大太阳,苦逼的一天;
程序员:继续上班,关我毛事;

使用Java语言具体实现过程:

1.发布者接口:

/*发布者接口,定义注册发布者,删除发布者,发布消息的接口*/
public interface IWeather {
    
    void addSubscriber(ISubscriber subscriber);
    
    void delSubscriber(ISubscriber subscriber);
    
    void publishInfo(String msg);
    
}

复制代码

2.发布者具体实现


import org.apache.log4j.Logger;

import java.util.ArrayList;
import java.util.List;

/*发布者具体实现类*/
public class WeatherServer implements IWeather {

    private static final Logger logger = Logger.getLogger(WeatherServer.class);

    /*用来保存注册了的所有订阅者*/
    private List<ISubscriber> subscribers = new ArrayList<ISubscriber>();

    public void addSubscriber(ISubscriber subscriber) {
        subscribers.add(subscriber);
        logger.info("a new subscriber is joining");
    }

    public void delSubscriber(ISubscriber subscriber) {
        subscribers.remove(subscriber);
        logger.info("a subscriber is leaving");
    }

    public void publishInfo(String msg) {
        for(ISubscriber subscriber : subscribers) subscriber.todoTomorrow(msg);
        logger.info(String.format("publish a msg: %s ",msg));
    }
}

复制代码

3.订阅者接口

/*订阅者接口,所有订阅WeatherServer 的订阅者都要实现该接口,定义了收到发布者消息之后做出反应的方法*/

public interface ISubscriber {
    void todoTomorrow(String msg);
}

复制代码

4.具体订阅者实现

import org.apache.log4j.Logger;
public class Farmer implements ISubscriber{
    private static final Logger logger = Logger.getLogger(Farmer.class);

    public void todoTomorrow(String msg) {

        if("rain".equals(msg)){
            logger.info("FARMER : a wonderful day!!!");
        }else if("sunny".equals(msg)){
            logger.info("FARMER : a enrich day!!!");
        }else {
            logger.info("FARMER : Spam messages");
        }
    }
}


public class Worker implements ISubscriber{
    private static final Logger logger = Logger.getLogger(Farmer.class);

    public void todoTomorrow(String msg) {

        if("rain".equals(msg)){
            logger.info("WORKER : a satisfied day!!!");
        }else if("sunny".equals(msg)){
            logger.info("WORKER : a terrible day!!!");
        }else {
            logger.info("WORKER : Spam messages");
        }
    }
}


public class Programmer implements ISubscriber{
    private static final Logger logger = Logger.getLogger(Farmer.class);

    public void todoTomorrow(String msg) {

        if("rain".equals(msg)){
            logger.info("PROGRAMMER :raining??? Irrelevant !!!");
        }else if("sunny".equals(msg)){
            logger.info("PROGRAMMER : sunny Irrelevant !!!");
        }else {
            logger.info("PROGRAMMER : Spam messages");
        }
    }
}

复制代码

5.测试类

 */
public class Boot {


    public static void main(String[] args){

        Farmer farmer = new Farmer();
        Worker worker = new Worker();
        Programmer programmer = new Programmer();

        WeatherServer weatherServer = new WeatherServer();

        weatherServer.addSubscriber(farmer);
        weatherServer.addSubscriber(worker);
        weatherServer.addSubscriber(programmer);

        weatherServer.publishInfo("rain");
    }
}

复制代码

6.运行结果:

有一个新的订阅者想要订阅天气预报的服务时,只需要调用weather的注册方法addSubscriber完成注册就能收到天气预报的信息。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值