Observer模式

观察者模式在软件编程中用的较多,我不善于表达我就用在网上看到的故事,和生活的例子说下基本思想 

故事:

   小雪是一个非常漂亮的女孩,漂亮的女孩总是有很多的追求者,而且追求者的队伍在不断的变动,随时有人进入这个队伍,也有人退出。男孩们追求女孩时总是表现出120%的关心,当小雪私自游玩时总是不断收到追求者询问小雪位置变动的消息,小雪也不胜其烦,但小雪是如此的一个善良的女孩,她总是打断自己正常的生活回复男孩们的消息。而男孩们由于要不断的关心小雪的位置变化也弄的精疲力竭,而且还影响正常的工作。在这样一个简单的故事场景中我们发现了什么?来看看小雪和男孩们的烦恼: 
  1.男孩们必须不断的询问小雪的位置变化,从而打断正常的工作; 
  2.小雪也要不断的接受男孩们的询问,有的时候小雪的位置并没有发生变化,还是要不断的回复男孩们的询问,也影响正常的工作。

  3.如果给各个男孩们回复问题的方式都不尽相同,小雪还要知道不同的回复方式,而且不断的有新的男孩们增加进来,还不知道未来有什么新的回复方式。

    看到这么多烦恼,我们创意无限的Nokia公司给小雪和男孩们提出了解决方案: 
    Nokia公司荣誉出品了一款带有GPRS功能的手机,该手机保存着一个订阅位置变化短信通知的电话列表,当该手机检测到位置发生变化就会向这个订阅列表里的所有手机发送短信。看到Nokia这个解决方案,男孩们和小雪都应该松一口气,他们各自都可以按照自己正常的生活习惯,只有状态发生变化时候各自才会进行通信

 

   观察者模式还可以用生活中一个例子来表达,就是从邮局订杂志。假如有一个叫 妮妮 的女孩在A邮局订了《时尚女孩》的杂志,又在B邮局订了《知音》杂志,并且告诉这两家邮局,如果杂志到了就给我打电话我自己来拿,然后邮局就在系统中注册下这个女孩姓名,电话等信息。妮妮剩下的就是等邮局的电话来取杂志了。如果杂志到了,邮局打电话给妮妮说,您的杂志到了,请到某某邮局来取(这相当于程序中把对象的引用——邮局名,传给观察者),如果只说您的杂志到了,请到邮局来取,妮妮怎么知道去哪个邮局拿杂志呀。

 

下面的程序模仿上面的情形,一个随机数产生对象和两个观察者,这两个观察者都在随机数产生对象那里注册了,意思说如果你产生了新的数字,就通知我一声。

 

结构图:

类说明
 名称 功能说明
 Observer 表示观察者的接口,要成为观察者必须实现此接口才行
 NumberGenerator 表示产生数值的抽象类
 RandomNumberGenerator 产生随机数的类,继承于NumberGenerator
 NumberObserver 数字观察者,会打印出变化的数字
 SymbolObserver 符号观察者,打印N 个符号,打印多少个符号,由接受到的数值确定

 

 

1.Observer

Java代码   收藏代码
  1. package com.pattern.observer;  
  2.   
  3. public interface Observer {  
  4.     public abstract void update(NumberGenerator generator);  
  5. }  

2.NumberGenerator

 

Java代码   收藏代码
  1. package com.pattern.observer;  
  2.   
  3. import java.util.ArrayList;  
  4. import java.util.Iterator;  
  5.   
  6. /** 
  7.  * @project JavaPattern 
  8.  * @author sunnylocus    
  9.  * @verson 1.0.0 
  10.  * @date   Aug 27, 2008 1:35:34 PM 
  11.  * @description 产生数值的抽象类 
  12.  */  
  13. public abstract class NumberGenerator {  
  14.     private ArrayList observers = new ArrayList();  //存储Observer  
  15.     /** 添加观察者*/  
  16.     public void addObserver(Observer observer) {  
  17.         observers.add(observer);  
  18.     }  
  19.     /** 删除观察者*/  
  20.     public void delObserver(Observer observer) {  
  21.         observers.remove(observer);  
  22.     }  
  23.     /** 通知所有观察者*/  
  24.     public void notifyObservers() {  
  25.         Iterator it = observers.iterator();  
  26.         while(it.hasNext()) {  
  27.             Observer o =(Observer) it.next();  
  28.             o.update(this);//this相当于上面提到的邮局名  
  29.         }  
  30.     }  
  31.     public abstract int getNumber();//获取数字  
  32.     public abstract void generate();//产生数字  
  33. }  

 

3.RandomNumberGenerator

Java代码   收藏代码
  1. package com.pattern.observer;  
  2.   
  3. import java.util.Random;  
  4.   
  5. /** 
  6.  * @project JavaPattern 
  7.  * @author sunnylocus    
  8.  * @verson 1.0.0 
  9.  * @date   Aug 27, 2008 1:48:03 PM 
  10.  * @description 用于产生随机数及通知观察者的类 
  11.  */  
  12. public class RandomNumberGenerator extends NumberGenerator{  
  13.     private Random random = new Random();//随机数产生器  
  14.     private int number;   //用于存放数字  
  15.   
  16.     public void generate() {  
  17.         for(int i=0 ; i < 5; i++) {  
  18.             number = random.nextInt(10);//产生10以内的随机数  
  19.             notifyObservers();  //有新产生的数字,通知所有注册的观察者  
  20.         }  
  21.     }  
  22.     /** 获得数字*/  
  23.     public int getNumber() {  
  24.         return number;  
  25.     }  
  26.       
  27. }  

 

4.NumberObserver

Java代码   收藏代码
  1. package com.pattern.observer;  
  2.   
  3. /** 以数字表示观察者的类*/  
  4. public class NumberObserver implements Observer{  
  5.     public void update(NumberGenerator generator) {  
  6.         System.out.println("NumberObserver:"+ generator.getNumber());  
  7.         try {  
  8.             Thread.sleep(1000 * 3); //为了能清楚的看到输出,休眠3秒钟。  
  9.         }catch(InterruptedException e) {  
  10.             e.printStackTrace();  
  11.         }  
  12.     }  
  13.       
  14. }  

 

5.SymbolObserver

Java代码   收藏代码
  1. package com.pattern.observer;  
  2.   
  3. /** 以符号表示观察者的类*/  
  4. public class SymbolObserver implements Observer{  
  5.     public void update(NumberGenerator generator) {  
  6.         System.out.print("SymbolObserver:");  
  7.         int count = generator.getNumber();  
  8.           
  9.         for(int i = 0 ; i < count; i ++) {  
  10.             System.out.print("*^_^*  ");  
  11.         }  
  12.         System.out.println("");  
  13.         try {  
  14.             Thread.sleep(1000 * 3);  
  15.         }catch(InterruptedException e){  
  16.             e.printStackTrace();  
  17.         }  
  18.     }  
  19. }  

 

6.Main(测试类)

Java代码   收藏代码
  1. package com.pattern.observer;  
  2.   
  3. public class Main {  
  4.     public static void main(String[] args) {  
  5.         //实例化数字产生对象  
  6.         NumberGenerator generator = new RandomNumberGenerator();  
  7.         //实例化观察者  
  8.         Observer observer1 = new NumberObserver();  
  9.         Observer observer2 = new SymbolObserver();  
  10.         //注册观察者  
  11.         generator.addObserver(observer1);  
  12.         generator.addObserver(observer2);  
  13.           
  14.         generator.generate(); //产生数字  
  15.     }  
  16. }  

 7.结果输出

 

设计思想:

     观察者模式定义了对象之间的一对多的依赖关系,当一个对象的状态发生改变时,所有它的依赖对象将被自动通知并更新


转自:http://sunnylocus.iteye.com/blog/233212

===============================华丽的分割线===================================================================

Observer模式的意图是“定义对象间的一种一对多的依赖关系(NumberGenerator和Observer就是一对多的关系),当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新”。从这段话里我们可以得到两个信息,如下: 
1, 观察者Observer(具体执行操作的对象,有多个,在上面的例子里就是Observer以及实现其的NumberObserver和SymbolObserver) 
2, 被观察者Observable(顾名思义是被观察的对象,如果该对象发生某些变化则通知观察者执行对应的操作,在上面的例子里就是NumberGenerator和RandomNumberGenerator) 


总结来说,所有观察者Observer都会实现一个共同的接口,这个接口中会定义一个方法,以供需要通知观察者时调用;观察者模式中被观察者Observable中会持有若干个Observer对象,一般是放在一个集合中,并且提供添加Observer、删除Observer和遍历Observer集合挨个调用Observer接口方法来通知各个Observer的方法;在使用的时候,初始化Observable时就会添加进各个需要通知的Observer,在调用完Observable主要业务后,调用遍历Observer的方法来通知各个Observer。


注:

在java.util包中已经提供了观察者模式的java类,包括接口Observer(包含update方法,在通知时调用)、类Observable(包含了添加、删除、通知Observer等方法)


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值