zoukankan      html  css  js  c++  java
  • Java Observer接口和Observable类实现观察者模式

    对于观察者模式,其实Java已经为我们提供了已有的接口和类。对于订阅者(Subscribe,观察者)Java为我们提供了一个接口,JDK源码如下:

    1 package java.util;
    2  
    3 public interface Observer {
    4     void update(Observable o, Object arg);
    5 }

    上述仅提供一个update方法用于接收通知者的通知做出相应改变。

    Java还为我们提供了一个通知者(Publish,发布者),JDK源码如下:

     1 package java.util;
     2 
     3 public class Observable {
     4     private boolean changed = false;  //是否改变状态
     5     private Vector obs;    //Vector利用同步方法来线程安全,线程安全在多线程情况下不会造成数据混乱
     6 
     7     public Observable() {
     8         obs = new Vector();
     9     }
    10 
    11     public synchronized void addObserver(Observer o) {
    12         if (o == null)
    13             throw new NullPointerException();
    14         if (!obs.contains(o)) {
    15             obs.addElement(o);
    16         }
    17     }
    18 
    19     public synchronized void deleteObserver(Observer o) {
    20         obs.removeElement(o);
    21     }
    22 
    23     public void notifyObservers() {
    24         notifyObservers(null);
    25     }
    26 
    27     public void notifyObservers(Object arg) {
    28         Object[] arrLocal;
    29 
    30         synchronized (this) {
    31             if (!changed)    //状态值未改变时返回,不通知
    32                 return;
    33             arrLocal = obs.toArray();  //将Vector转换成数组
    34             clearChanged();    //重置状态
    35         }
    36 
    37         for (int i = arrLocal.length-1; i>=0; i--)
    38             ((Observer)arrLocal[i]).update(this, arg);
    39     }
    40 
    41     public synchronized void deleteObservers() {
    42         obs.removeAllElements();
    43     }
    44 
    45     protected synchronized void setChanged() {
    46         changed = true;
    47     }
    48 
    49     protected synchronized void clearChanged() {
    50         changed = false;
    51     }
    52 
    53     public synchronized boolean hasChanged() {
    54         return changed;
    55     }
    56 
    57     public synchronized int countObservers() {
    58         return obs.size();
    59     }
    60 }

    首先,使用Vector,Vector相比于ArrayList来说,它是线程安全的。其次,在添加和删除观察者时对两个方法使用了synchronized关键字,这都是在为多线程考虑。确实Java源码并不是那么可怕,它同样也是由一些最简单最基础的组合而来。

    接下来看看是如何利用Java提供的接口和方法来实现观察者模式。

    Observable:

     1 public class Publish extends Observable {
     2 
     3     private String data = "";
     4 
     5     public String getData() {
     6         return data;
     7     }
     8 
     9     public void setData(String data) {
    10         if (!this.data.equals(data)) {
    11             this.data = data;
    12             setChanged();    //改变通知者的状态
    13         }
    14         notifyObservers();    //调用父类Observable方法,通知所有观察者
    15     }
    16 }

    Observer:
     1 public class Subscribe implements Observer {
     2 
     3     public Subscribe(Observable o) {
     4         //将该观察者放入待通知观察者里
     5         o.addObserver(this);
     6     }
     7 
     8     @Override
     9     public void update(Observable o, Object arg) {
    10         System.out.println("收到了消息:" + ((Publish)o).getData());
    11     }
    12 }

    测试:

    1 public class ObserverTest {
    2 
    3     public static void main(String[] args) {
    4         Publish publish = new Publish();
    5         Subscribe subscribe = new Subscribe(publish);
    6         publish.setData("发布一个消息");
    7     }
    8 }

    运行结果:

    收到了消息:发布一个消息

    这样就利用了Java给我们提供的Observer接口和Observable类实现了观察者模式。

  • 相关阅读:
    orm 对象关系映射 指 表与类之间的映射 # 40
    事务 视图 触发器 函数 (内置) 存储过程 流程控制 索引 # 39
    exist 存在 Python操作mysql pymysql sql注入问题 # 38
    基本查询语句与方法 多表查询 # 37
    外键 #36
    存储引擎 索引 数据类型 约束条件 # 35
    mysql安装 登录 修改密码 库,表,记录(增删改查) # 34
    进程池和线程池 协程 # 33
    GIL全局解释器锁
    # 并发编程 -进程理论-进程的方法
  • 原文地址:https://www.cnblogs.com/UniqueColor/p/10610134.html
Copyright © 2011-2022 走看看