转载

深入JDK源码之Observer接口和Observable类实现观察者模式

何为观察者模式

观察者模式(有时又被称为发布/订阅模式)是软体 设计模式 的一种。在此种模式中, 一个目标物件管理所有相依于它的观察者物件,并且在它本身的状态改变时主动发出通知 。这通常透过呼叫各观察者所提供的方法来实现。此种模式通常被用来事件处理系统。

使用它有何好处

观察者模式(Observer)完美的将观察者和被观察的对象分离开。举个例子,用户界面可以作为一个观察者,业务数据是被观察者,用户界面观察业务数据的变化,发现数据变化后,就显示在界面上。 面向对象设计的一个原则是:系统中的每个类将重点放在某一个功能上,而不是其他方面 。一个对象只做一件事情,并且将他做好。观察者模式在模块之间划定了清晰的界限,提高了应用程序的可维护性和重用性。

实现方式

观察者模式有很多实现方式,从根本上说,该模式必须包含两个角色: 观察者和被观察者 。在刚才的例子中,业务数据是被观察对象,用户界面是观察者。观察者和被观察者之间存在“观察”的逻辑关联,当被观察者发生改变的时候,观察者就会观察到这样的变化,并且做出相应的响应。如果在用户界面、业务数据之间使用这样的观察过程,可以确保界面和数据之间划清界限,假定应用程序的需求发生变化,需要修改界面的表现,只需要重新构建一个用户界面,业务数据不需要发生变化。

JDK中的Observer接口和Observable类实现

1. 在Observer接口源代码中只声明一个update()方法

public interface Observer {     //这个方法被每当观测目标被改变了,让被观察者调用     void update(Observable o, Object arg); }

该方法让被观察者的对象当改变时调用这个方法。实现这个接口的类,可以作为观察者。

2. Observable类,被观察者,消息的发出者,继承此类可以扩展

package java.util; public class Observable {     private boolean changed = false;     //观察者的集合     private Vector obs;      /** Construct an Observable with zero Observers. */     public Observable() {         obs = new Vector();     }      public synchronized void addObserver(Observer o) {         if (o == null)             throw new NullPointerException();         if (!obs.contains(o)) {             obs.addElement(o);         }     }      public synchronized void deleteObserver(Observer o) {         obs.removeElement(o);     }      public void notifyObservers() {         notifyObservers(null);     }      //通知所有订阅此主题的观察者对象     public void notifyObservers(Object arg) {         Object[] arrLocal;         //同步代码块         synchronized (this) {             //若主题没有改变,返回             if (!changed)                 return;             arrLocal = obs.toArray();             clearChanged();         }          for (int i = arrLocal.length-1; i>=0; i--)             //通知观察者,调用观察者的update()方法             ((Observer)arrLocal[i]).update(this, arg);     }      //清空所有观察此主题的观察者     public synchronized void deleteObservers() {         obs.removeAllElements();     }      //主题改变     protected synchronized void setChanged() {         changed = true;     }      //清除改变     protected synchronized void clearChanged() {         changed = false;     }      //判断主题是否改变     public synchronized boolean hasChanged() {         return changed;     }      //返回观察者的数量     public synchronized int countObservers() {         return obs.size();     } }

观察者模式Demo实例

package com.king.pattem.observer;  import java.util.*;  class TieDaoBu extends Observable {    // 表示铁道部可以被观察     private String message;// 官方消息      public TieDaoBu(String message) {         this.message = message;     }      public String getMessage() {         return this.message;     }      public void setMessage(String message) {         // 每一次修改的时候都应该引起观察者的注意         System.out.println("****************************************");         System.out.println(new Date() + "官方发布消息为:" + message);         super.setChanged();    // 设置变化点         super.notifyObservers(message);// 通知所有观察者         this.message = message;     } }  class TieDaoBuObserver implements Observer {     private String name;      public TieDaoBuObserver(String name) {    // 设置每一个观察者的名字         this.name = name;     }      public void update(Observable o, Object arg) {         System.out.print(this.name + " 官方消息更改为:");         System.out.println(arg.toString());     } }  public class JDKObserverDemo {     public static void main(String args[]) {         TieDaoBu h = new TieDaoBu("温州火车出轨体现了我国高铁世界技术水平领先,和谐社会和谐号出事了");         TieDaoBuObserver hpo1 = new TieDaoBuObserver("媒体A");         TieDaoBuObserver hpo2 = new TieDaoBuObserver("屁民B");         TieDaoBuObserver hpo3 = new TieDaoBuObserver("日本韩国嘲笑者C");         h.addObserver(hpo1);         h.addObserver(hpo2);         h.addObserver(hpo3);          h.setMessage("搜救结束,经生命探测仪发现没有生命迹象");    // 修改官方消息          h.setMessage("搜救结束了,还发现了一名2岁的女孩,真是奇迹");    // 修改官方消息          h.setMessage("35一个神奇的数字,动车相撞35人死亡,河南平顶山矿难35人死亡," +                 "重庆暴雨35人死亡,云南大雨35人死亡。" +                 "为什么死亡人数控制在36人以内?" +                 "超过36人市委书记级别的要撤职,所以一开始发生就注定了死亡人数不会超过36。" +                 "而事实上,我在查看国外报纸报道," +                 "华尔街日报说,这次动车事故,其中有47人死亡,200余人受伤");    // 修改官方消息     } }
原文  http://www.importnew.com/19568.html
正文到此结束
Loading...