zoukankan      html  css  js  c++  java
  • 观察者模式

    一、概念

      观察者模式,又被称为发布—订阅模式、源—收听者模式,是软件设计模式的一种。在此种模式中,一个目标物件管理所有相依与它的观察者物件,并且在它本身的状态改变时主动发出同时,此种模式通常被用来实现事件处理系统。

    二、观察者模式组成

      观察者模式一般包含以下四种角色:

      抽象主题角色Watched:把所有对观察者对象的引用保存在一个集合中,每个抽象主题角色都可以有任意数量的观察者。抽象主题提供一个接口,可以增加和删除观察这角色。

      抽象观察者角色Watcher:为所有具体的观察者定义一个接口,在得到主题的通知更新自己。

      具体主题角色ConcreteWatched:在具体主题内部状态改变时,给所有登记过的观察者发出通知。

      具体观察者角色ConcreteWatcher:该角色实现抽象观察者角色所要求的更新接口,以便使自身的状态与主题的状态相协调。

    三、观察者模式实现方式

      观察者模式有很多实现方式,从根本上说,该模式必须包含两个角色,观察者和被观察对象。比较直观的一种是使用“注册”—“通知”—“撤销通知”的形式。

      观察者将自己注册到被观察对象中,被观察者将观察者放在一个容器中。

      被观察者对象发生某种变化时,从容器中得到所有被注册过的观察者,将变化通知观察者。

      观察者将自己注册到被观察者的容器时,被观察者不应该过问观察者的具体类型,而是应该使用观察者的接口。这样的好处是:假定程序还有别的观察者,那么只要这个观察者也是相同的接口即可。一个被观察者可以对应多个观察者,当被观察者发生变化的时候,他可以将消息一一通知给所有的观察者。如下图所示:

      

      具体到一个应用场景就是:读者订阅新闻,每一个读者都可以通过regist来订阅新闻,通过remove来取消订阅,当有新闻到来时,使用sendNews来通知每一位订阅的读者来收取新闻。

      下面用代码来实现上面的实例:

      抽象主题角色Watched:新闻接口,定义订阅、取消订阅和推送新闻三个方法

    public interface Watched
    {
        //订阅新闻
        void registeSubscriber(Watcher f_subscriber);
        //取消订阅
        void removeSubscriber(Watcher f_subscriber);
        //推送新闻
        void sendNews();
    }

      抽象观察者角色Watcher:读者接口,定义了获取新闻的方法

    public interface Watcher
    {
        //获取新闻
        void updateNews();
    }

      具体主题角色ConcreteWatched,即被观察者,实现了Watched接口:

    import java.util.ArrayList;
    import java.util.List;
    
    /*
     * 具体被观察者
     * 新闻发布者
     */
    public class ConcreteWatched implements Watched
    {
        private List<Watcher> subList=new ArrayList<Watcher>();
        @Override
        public void registeSubscriber(Watcher f_subscriber)
        {
            subList.add(f_subscriber);    
        }
        @Override
        public void removeSubscriber(Watcher f_subscriber)
        {
            subList.remove(f_subscriber);        
        }
        @Override
        public void sendNews()
        {
         System.out.println("开始本轮新闻推送!");
    for (Watcher watcher : subList) { watcher.updateNews(); } System.out.println("本次新闻推送结束!"); } }

      具体观察者角色ConcreteWatcher,即观察者对象

    /*
     * 具体的观察者
     * 读者
     */
    public class ConcreteWatcher implements Watcher
    {
        private String username;
            
        public ConcreteWatcher(String username)
        {
            super();
            this.username = username;
        }
        @Override
        public void updateNews()
        {
            System.out.println(username+"获取到最新的新闻了!");
            
        }
    }

      下面是一个测试用例:

    import org.junit.Test;
    
    public class WatchTest
    {
        @Test
        public void test()
        {
            ConcreteWatched concreteWatched=new ConcreteWatched();
            Watcher watcher1=new ConcreteWatcher("读者A");
            Watcher watcher2=new ConcreteWatcher("读者B");
            Watcher watcher3=new ConcreteWatcher("读者C");
            
            //订阅新闻
            concreteWatched.registeSubscriber(watcher1);
            concreteWatched.registeSubscriber(watcher2);
            concreteWatched.registeSubscriber(watcher3);
            
            //推送新闻
            concreteWatched.sendNews();
            //读者A取消订阅
            concreteWatched.removeSubscriber(watcher1);
    
            //再次推送新闻
            concreteWatched.sendNews();
        }
    }

      执行结果:

      

    四、参考资料

      1、http://www.cnblogs.com/mengdd/archive/2013/02/07/2908929.html

  • 相关阅读:
    【推荐】微服务分布式企业框架Springmvc+mybatis+shiro+Dubbo+ZooKeeper+Redis
    将WCF寄宿在托管的Windows服务中
    剑指Offer
    剑指Offer
    Entity Framework 无法对没有主键的视图映射实体的解决办法
    设置Sql Agent运行Job时的执行账户
    IE中Keep-Alive机制引起的错误
    Ajax错误 “SCRIPT7002: XMLHttpRequest: 网络错误 0x2ef3, 由于出现错误 00002ef3 而导致此项操作无法完成” 的归纳总结
    在IIS站点中Adomd.net集成认证账号问题
    出现“不能执行已释放的Script代码”错误的原因及解决办法
  • 原文地址:https://www.cnblogs.com/xujian2014/p/5557612.html
Copyright © 2011-2022 走看看