随笔-295  评论-26  文章-1  trackbacks-0
java 观察者模式
2009-09-10 09:51

观察者模式可以参考邮件订阅的例子

邮件订阅设计到2个主要角色,一个是订阅者(观察者),一个是发布者

发布者可以拥有一个观察者的集合,可以添加,删除观察者,当发布者发布一个新的消息时,要邮件通知观察者集合中的每一个。

所以,发布者实现的接口至少应具备三个方法,即注册观察者,注销观察者,通知观察者。

通知有两种方式,一种是推(push),一种是拉(pull).

推,即发布者通过调用观察者提供的接口,来告之所有的变动(比如新增一个消息),主动推送给观察者。

拉,及观察者可以按需提取所要接受的数据,而不是全盘接受,主要体现在程序自己主动调用观察者的传值接口,而区别于推由发布者来调用,此时,需要这个借口中的参数包含发布者对象,让程序知道扫描变动来自于哪个发布者。

JDK中内置了观察者模式,位于java.util包中,一个接口Observer,一个类Observable,将这两个类组合起来使用,既可以推,又可以拉。

一个普通的类,如果继承了Observable,就成了一个发布者,实现了Observer接口,就成了观察者。

JDK中的观察者模式有一个弊端,就是发布者需要继承一个类,而不是实现一个接口,如果需要成为发布者的类已经集成了一个类,就不能再继承Observable了。这时候,就需要自己来设计一种观察者模式了。

JDK 中Swing包中,大量运用了观察者模式,所有的组件都继承了JComponent,这就是一个发布者,它里面包含一个监听器的集合:EventListenerList,用户可以自定义一个监听器,然后添加到一个组件中的时候,该组件就会把这个监听器注册到 EventListenerList中,相当于添加了一个发布者,当用户对组件做出反映时,所有的监听器(发布者)都会收到信息并作出反应。

网上商店中的商品在名称、价格发生变化时,必须自动通知会员,Java的API为我们提供了
   Observer接口和Observable类来实现所谓观察者模式。Observable(可观察者)类允许在自身发生改变时,通知其它对象(实现接口Observer,观察者)。


   下面是一个可观察者(产品类):
import java.util.*;
public class product extends Observable{
   private String name;////产品名
   private float price;////价格

   public String getName(){ return name;}
   public void setName(String name){
   this.name=name;
   ////设置变化点
   setChanged();
   notifyObservers(name);////通知观察者

   }   

   public float getPrice(){ return price;}
   public void setPrice(float price){
   this.price=price;
   ////设置变化点
   setChanged();
   notifyObservers(new Float(price));

   }

   ////以下可以是数据库更新 插入命令.
   public void saveToDb(){
   System.out.println("saveToDb");
     }

}

下面是两个观察者:
import java.util.*;
public class NameObserver implements Observer{

   private String name=null;
   public void update(Observable obj,Object arg){
     if (arg instanceof String){
     name=(String)arg;
     ////产品名称改变值在name中
     System.out.println("NameObserver :name changet to "+name);

     }

        }
   }

import java.util.*;
public class PriceObserver implements Observer{
   private float price=0;
   public void update(Observable obj,Object arg){
     if (arg instanceof Float){

     price=((Float)arg).floatValue();
  
     System.out.println("PriceObserver :price changet to "+price);

     }


   }

}
下面是测试类:
public class Test {

   public static void main(String args[]){
      Product product=new Product();
      NameObserver nameobs=new NameObserver();
      PriceObserver priceobs=new PriceObserver();

      ////加入观察者
      product.addObserver(nameobs);
      product.addObserver(priceobs);

      product.setName("applet");
      product.setPrice(9.22f);

   }
}
运行结果:
C:\java>java    Test
NameObserver :name changet to applet



大盘预测 国富论
posted on 2009-10-22 10:16 华梦行 阅读(515) 评论(0)  编辑  收藏

只有注册用户登录后才能发表评论。


网站导航: