一、什么是订阅发布模式
订阅发布模式定义了一种 一对多 的依赖关系,一指发布者,比如某个主题对象,多指订阅者,依赖是订阅者对发布者的依赖;多个订阅者同时监听一个主题对象。当发布者即主题对象的状态发生变化时,会将该变化通知给订阅者,订阅者据此更新自己的状态。是设计模式中用的比较多的一种。
二、为什么使用订阅发布模式
订阅发布模式旨在降低系统不同模块之间的耦合度。一个复杂的系统一般要划分为好多不同的类,这些对象之间并不独立,存在各种协作关系,这就需要维护对象之间的一致性,这给系统的维护、重用和扩展带来不便,严重限制了系统的灵活性。这样就可以把主动变化因素抽象出来形成发布者,而其他依赖者抽象为订阅者,使他们可以独立的被复用和维护。双发都只依赖与抽象,而不是依赖于具体实现。比如发布者新增一个订阅者时只需要发布者注册一下,订阅者新增依赖时,只需要将自己注册到新的发布者即可。
三、一个例子:
有一个天气预报系统,负责通知明天的天气情况。有几个角色订阅了该系统,他们根据明天的天气情况安排明天的事务。这几个角色分别为农夫、建筑工人、程序员。
下雨天
农夫喜出望外:又可以好好歇一天了,让庄稼庄稼好好成长吧; 建筑工人欣喜万分:完美,睡到天昏地暗; 程序员:继续上班,关我毛事;日晴万里
农夫感慨:即将迎接充实的一天,收获满满; 建筑工人牢骚:这么大太阳,苦逼的一天; 程序员:继续上班,关我毛事;使用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 Listsubscribers = new ArrayList (); 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完成注册就能收到天气预报的信息。