唯一浩哥

架构之路----Java有不少事

Java设计模式之《观察者模式》及应用场景

原创作品,可以转载,但是请标注出处地址:http://www.cnblogs.com/V1haoge/p/6513651.html

  观察者模式,又可以称之为发布-订阅模式,观察者,顾名思义,就是一个监听者,类似监听器的存在,一旦被观察/监听的目标发生的情况,就会被监听者发现,这么想来目标发生情况到观察者知道情况,其实是由目标将情况发送到观察者的。

  观察者模式多用于实现订阅功能的场景,例如微博的订阅,当我们订阅了某个人的微博账号,当这个人发布了新的消息,就会通知我们。

  现在我们举一个类似的情况,并使用代码来实现,为大家提供一个比较明显的认识。

  警察在找到嫌犯的时候,为了找到幕后主使,一般都会蹲点监察,这里我有三名便衣警察来蹲点监察2名嫌犯,三名便衣分别是:张昊天、石破天、赵日天,两名嫌犯是:大熊与黑狗,详见代码:

观察者接口:Observer

1 public interface Observer {
2     void update(String message,String name);
3 }

定义三名便衣观察者:Bianyi1、Bianyi2、Bianyi3

 1 /**
 2  * 便衣警察张昊天
 3  */
 4 public class Bianyi1 implements Observer {
 5     //定义姓名
 6     private String bname = "张昊天";
 7     @Override
 8     public void update(String message,String name) {
 9         System.out.println(bname+":"+name+"那里有新情况:"+ message);
10     }
11 }
12 
13 /**
14  * 便衣警察石破天
15  */
16 public class Bianyi2 implements Observer {
17     //定义姓名
18     private String bname = "石破天";
19     @Override
20     public void update(String message,String name) {
21         System.out.println(bname+":"+name+"那里有新情况:"+ message);
22     }
23 }
24 
25 /**
26  * 便衣警察赵日天
27  */
28 public class Bianyi3 implements Observer {
29     //定义姓名
30     private String bname = "赵日天";
31     @Override
32     public void update(String message,String name) {
33         System.out.println(bname+":"+name+"那里有新情况:"+ message);
34     }
35 }

目标接口:Huairen

1 public interface Huairen {
2     //添加便衣观察者
3     void addObserver(Observer observer);
4     //移除便衣观察者
5     void removeObserver(Observer observer);
6     //通知观察者
7     void notice(String message);
8 }

定义两个嫌疑犯:XianFan1、XianFan2

 1 import java.util.*;
 2 /**
 3  * 嫌犯大熊
 4  */
 5 public class XianFan1 implements Huairen {
 6     //别称
 7     private String name = "大熊";
 8     //定义观察者集合
 9     private List<Observer> observerList = new ArrayList<Observer>();
10     //增加观察者
11     @Override
12     public void addObserver(Observer observer) {
13         if(!observerList.contains(observer)){
14             observerList.add(observer);
15         }
16     }
17     //移除观察者
18     @Override
19     public void removeObserver(Observer observer) {
20         if(observerList.contains(observer)){
21             observerList.remove(observer);
22         }
23     }
24     //通知观察者
25     @Override
26     public void notice(String message) {
27         for(Observer observer:observerList){
28             observer.update(message,name);
29         }
30     }
31 }
32 
33 import java.util.*;
34 /**
35  * 嫌犯黑狗
36  */
37 public class XianFan2 implements Huairen {
38     //别称
39     private String name = "黑狗";
40     //定义观察者集合
41     private List<Observer> observerList = new ArrayList<Observer>();
42     //增加观察者
43     @Override
44     public void addObserver(Observer observer) {
45         if(!observerList.contains(observer)){
46             observerList.add(observer);
47         }
48     }
49     //移除观察者
50     @Override
51     public void removeObserver(Observer observer) {
52         if(observerList.contains(observer)){
53             observerList.remove(observer);
54         }
55     }
56     //通知观察者
57     @Override
58     public void notice(String message) {
59         for(Observer observer:observerList){
60             observer.update(message,name);
61         }
62     }
63 }

测试类:Clienter

 1 public class Clienter {
 2     public static void main(String[] args) {
 3         //定义两个嫌犯
 4         Huairen xf1 = new XianFan1();
 5         Huairen xf2 = new XianFan2();
 6         //定义三个观察便衣警察
 7         Observer o1 = new Bianyi1();
 8         Observer o2 = new Bianyi2();
 9         Observer o3 = new Bianyi3();
10         //为嫌犯增加观察便衣
11         xf1.addObserver(o1);
12         xf1.addObserver(o2);
13         xf2.addObserver(o1);
14         xf2.addObserver(o3);
15         //定义嫌犯1的情况
16         String message1 = "又卖了一批货";
17         String message2 = "老大要下来视察了";
18         xf1.notice(message1);
19         xf2.notice(message2);
20     }
21 }

测试结果:

张昊天:大熊那里有新情况:又卖了一批货
石破天:大熊那里有新情况:又卖了一批货
张昊天:黑狗那里有新情况:老大要下来视察了
包拯:黑狗那里有新情况:老大要下来视察了

  通过上面的实例可以很明显的看出,观察者模式的大概模型,关键是什么呢?

  关键点:

    1、针对观察者与被观察者分别定义接口,有利于分别进行扩展。

    2、重点就在被观察者的实现中:

      (1)定义观察者集合,并定义针对集合的添加、删除操作,用于增加、删除订阅者(观察者)

      (2)定义通知方法,用于将新情况通知给观察者用户(订阅者用户)

    3、观察者中需要有个接收被观察者通知的方法。

  如此而已!

  观察者模式定义的是一对多的依赖关系,一个被观察者可以拥有多个观察者,并且通过接口对观察者与被观察者进行逻辑解耦,降低二者的直接耦合。

  如此这般,想了一番之后,突然发现这种模式与桥接模式有点类似的感觉。

  桥接模式也是拥有双方,同样是使用接口(抽象类)的方式进行解耦,使双方能够无限扩展而互不影响,其实二者还是有者明显的区别:

    1、主要就是使用场景不同,桥接模式主要用于实现抽象与实现的解耦,主要目的也正是如此,为了双方的自由扩展而进行解耦,这是一种多对多的场景。观察者模式侧重于另一方面的解耦,侧重于监听方面,侧重于一对多的情况,侧重于一方发生情况,多方能获得这个情况的场景。

    2、另一方面就是编码方面的不同,在观察者模式中存在许多独有的内容,如观察者集合的操作,通知的发送与接收,而在桥接模式中只是简单的接口引用。

 


同系列文章:

posted @ 2017-03-07 11:03  唯一浩哥  阅读(29909)  评论(5编辑  收藏  举报