Java设计模式开发中使用观察者模式的实例教程

 更新时间:2016年04月24日 09:53:43   作者:匆忙拥挤repeat  
这篇文章主要介绍了Java设计模式开发中使用观察者模式的实例教程,松耦合和逻辑清晰的消息监听是观察者模式的大特色,需要的朋友可以参考下

观察者模式是软件设计模式中的一种,使用也比较普遍,尤其是在GUI编程中。关于设计模式的文章,网络上写的都比较多,而且很多文章写的也不错,虽然说有一种重复早轮子的嫌疑,但此轮子非彼轮子,侧重点不同,思路也不同,讲述方式也不近相同。
关键要素

主题:

主题是观察者观察的对象,一个主题必须具备下面三个特征。

  • 持有监听的观察者的引用
  • 支持增加和删除观察者
  • 主题状态改变,通知观察者

观察者:

当主题发生变化,收到通知进行具体的处理是观察者必须具备的特征。

为什么要用这种模式

这里举一个例子来说明,牛奶送奶站就是主题,订奶客户为监听者,客户从送奶站订阅牛奶后,会每天收到牛奶。如果客户不想订阅了,可以取消,以后就不会收到牛奶。

松耦合

观察者增加或删除无需修改主题的代码,只需调用主题对应的增加或者删除的方法即可。
主题只负责通知观察者,但无需了解观察者如何处理通知。举个例子,送奶站只负责送递牛奶,不关心客户是喝掉还是洗脸。
观察者只需等待主题通知,无需观察主题相关的细节。还是那个例子,客户只需关心送奶站送到牛奶,不关心牛奶由哪个快递人员,使用何种交通工具送达。

Java实现观察者模式
1. Java自带的实现
类图

201642494900291.jpg (806×524)

/** 
 * 观察目标 继承自 java.util.Observable 
 * @author stone 
 * 
 */ 
public class UpdateObservable extends Observable { 
   
  private int data; 
   
  public UpdateObservable(Observer observer) { 
    addObserver(observer); 
    /* 
     * add other observer 
     */ 
  } 
   
  public int getData() { 
    return data; 
  } 
   
  public void setData(int data) { 
    if (data != this.data) { 
      this.data = data; 
      setChanged(); //标记 改变, 只有标记后才能通知到 
      notifyObservers(); //通知 
    } 
     
  } 
 
  @Override 
  public synchronized void addObserver(Observer o) { 
    super.addObserver(o); 
  } 
 
  @Override 
  public synchronized void deleteObserver(Observer o) { 
    super.deleteObserver(o); 
  } 
 
  @Override 
  public void notifyObservers() { 
    super.notifyObservers(); 
  } 
 
  @Override 
  public void notifyObservers(Object arg) { 
    super.notifyObservers(arg); 
  } 
 
  @Override 
  public synchronized void deleteObservers() { 
    super.deleteObservers(); 
  } 
 
  @Override 
  protected synchronized void setChanged() { 
    super.setChanged(); 
  } 
 
  @Override 
  protected synchronized void clearChanged() { 
    super.clearChanged(); 
  } 
 
  @Override 
  public synchronized boolean hasChanged() { 
    return super.hasChanged(); 
  } 
 
  @Override 
  public synchronized int countObservers() { 
    return super.countObservers(); 
  } 
   
} 

/** 
 * 观察者 实现 java.util.Observer接口 
 * @author stone 
 * 
 */ 
public class UpdateObserver implements Observer { 
 
  @Override 
  public void update(Observable o, Object arg) { 
    System.out.println("接收到数据变化的通知:"); 
    if (o instanceof UpdateObservable) { 
      UpdateObservable uo = (UpdateObservable) o; 
      System.out.print("数据变更为:" + uo.getData()); 
    } 
  } 
 
} 

2. 自定义的观察模型
类图

201642494952492.jpg (968×676)

/** 
 * 抽象观察者  Observer 
 * 观察 更新     
 * @author stone 
 * 
 */ 
public interface IWatcher { 
  /* 
   * 通知接口: 
   * 1. 简单通知 
   * 2. 观察者需要目标的变化的数据,那么可以将目标用作参数, 见Java的Observer和Observable 
   */ 
// void update(IWatched watched); 
   
  void update(); 
   
} 

/** 
 * 抽象目标 Subject 
 * 提供注册和删除观察者对象的接口, 及通知观察者进行观察的接口 
 * 及目标 自身被观察的业务的接口 
 * @author stone 
 * 
 */ 
public interface IWatchedSubject { 
   
  public void add(IWatcher watch); 
 
  public void remove(IWatcher watch); 
   
  public void notifyWhatchers(); 
   
  public void update();//被观察业务变化的接口 
} 

/** 
 * 具体观察者    Concrete Observer 
 * 
 * @author stone 
 * 
 */ 
public class UpdateWatcher implements IWatcher { 
 
  @Override 
  public void update() { 
    System.out.println(this + "观察到:目标已经更新了"); 
  } 
 
} 


/** 
 * 具体目标角色  Concrete Subject 
 * @author stone 
 * 
 */ 
public class UpdateWatchedSubject implements IWatchedSubject { 
  private List<IWatcher> list; 
   
  public UpdateWatchedSubject() { 
    this.list = new ArrayList<IWatcher>(); 
  } 
   
  @Override 
  public void add(IWatcher watch) { 
    this.list.add(watch); 
  } 
 
  @Override 
  public void remove(IWatcher watch) { 
    this.list.remove(watch); 
  } 
 
  @Override 
  public void notifyWhatchers() { 
    for (IWatcher watcher : list) { 
      watcher.update(); 
    } 
  } 
   
  @Override 
  public void update() { 
    System.out.println("目标更新中...."); 
    notifyWhatchers(); 
  } 
 
} 


监听器是观察者的一种实现:
类图

201642495137152.jpg (780×502)

/** 
 * 监听 用户在注册后 
 * @author stone 
 * 
 */ 
public interface IRegisterListener { 
  void onRegistered(); 
} 

/** 
 * 监听 当用户登录后 
 * @author stone 
 * 
 */ 
public interface ILoginListener { 
  void onLogined(); 
} 

/* 
 * 监听器 是观察者模式的一种实现 
 * 一些需要监听的业务接口上添加 监听器,调用监听器的相应方法,实现监听 
 */ 
public class User { 
   
  public void register(IRegisterListener register) { 
    /* 
     * do ... register 
     */ 
    System.out.println("正在注册中..."); 
    //注册后 
    register.onRegistered(); 
  } 
 
  public void login(ILoginListener login) { 
    /* 
     * do ... login 
     */ 
    System.out.println("正在登录中..."); 
    //登录后 
    login.onLogined(); 
  } 
 
} 


/** 
 * 观察者(Observer)模式 行为型模式 
 *  观察者模式定义了一种一对多的依赖关系,让多个观察者对象同时观察某一个目标对象。 
 *  这个目标对象在状态上发生变化时,会通知所有观察者对象,让它们能够自动更新自己 
 * 目标对象中需要有添加、移除、通知 观察者的接口 
 *  
 * @author stone 
 */ 
public class Test { 
   
  public static void main(String[] args) { 
    /* 
     * 使用Java自带的Observer接口和Observable类 
     */ 
    UpdateObservable observable = new UpdateObservable(new UpdateObserver()); 
    observable.setData(99); 
    System.out.println(""); 
    System.out.println(""); 
    /* 
     * 自定义的观察者模型 
     */ 
    IWatchedSubject watched = new UpdateWatchedSubject(); 
    watched.add(new UpdateWatcher()); 
    watched.add(new UpdateWatcher()); 
    watched.update(); 
    System.out.println(""); 
     
    /* 
     * 子模式-监听器 
     */ 
     
    User user = new User(); 
    user.register(new IRegisterListener() { 
       
      @Override 
      public void onRegistered() { 
        System.out.println("监听到注册后。。。"); 
      } 
    }); 
    user.login(new ILoginListener() { 
       
      @Override 
      public void onLogined() { 
        System.out.println("监听到登录后。。。"); 
      } 
    }); 
     
  } 
} 


打印

接收到数据变化的通知: 
数据变更为:99 
 
目标更新中.... 
observer.UpdateWatcher@457471e0观察到:目标已经更新了 
observer.UpdateWatcher@5fe04cbf观察到:目标已经更新了 
 
正在注册中... 
监听到注册后。。。 
正在登录中... 
监听到登录后。。。 

相关文章

  • Java内存模型及其作用详解

    Java内存模型及其作用详解

    在现代计算机系统中,尤其是多处理器架构下,每个处理器都有自己的高速缓存,而主内存(RAM)是所有处理器共享的数据存储区域,这篇文章主要介绍了深入理解Java内存模型及其作用,需要的朋友可以参考下
    2024-04-04
  • Java实现蓝桥杯G将军的示例代码

    Java实现蓝桥杯G将军的示例代码

    这篇文章主要介绍了Java实现蓝桥杯G将军的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-02-02
  • java 汉诺塔Hanoi递归、非递归(仿系统递归)和非递归规律 实现代码

    java 汉诺塔Hanoi递归、非递归(仿系统递归)和非递归规律 实现代码

    汉诺塔(Hanoi) 算法Java实现。通过三个函数,分别对Hanoi进行递归、非递归和非递归规律实现。
    2013-05-05
  • SpringBoot自动配置原理分析

    SpringBoot自动配置原理分析

    这篇文章主要介绍了SpringBoot自动配置原理分析,SpringBoot是我们经常使用的框架,那么你能不能针对SpringBoot实现自动配置做一个详细的介绍。如果可以的话,能不能画一下实现自动配置的流程图。牵扯到哪些关键类,以及哪些关键点
    2022-08-08
  • 解决JavaWeb读取本地json文件以及乱码的问题

    解决JavaWeb读取本地json文件以及乱码的问题

    今天小编就为大家分享一篇解决JavaWeb读取本地json文件以及乱码的问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-06-06
  • Java 正则表达式入门详解(基础进阶)

    Java 正则表达式入门详解(基础进阶)

    最近看到很多同学想要学习java正则表达式的一些知识,那么脚本之家小编就为大家介绍一下,其实正则表达式实用性很强,处理大幅文字的时候都需要用得到,语法也大同小异
    2017-10-10
  • 利用Spring Boot和JPA创建GraphQL API

    利用Spring Boot和JPA创建GraphQL API

    这篇文章主要介绍了利用Spring Boot和JPA创建GraphQL API,GraphQL既是API查询语言,也是使用当前数据执行这些查询的运行时,下文更多相关内容介绍需要的小伙伴可以参考一下
    2022-04-04
  • Spring Boot使用Hibernate-Validator校验参数时的长度校验方法详解

    Spring Boot使用Hibernate-Validator校验参数时的长度校验方法详解

    这篇文章主要给大家介绍了关于Spring Boot使用Hibernate-Validator校验参数时的长度校验方法的相关资料,在Spring Boot中可以使用Spring Boot Validation来对参数名称进行校验,需要的朋友可以参考下
    2023-08-08
  • Redis使用RedisTemplate模板类的常用操作方式

    Redis使用RedisTemplate模板类的常用操作方式

    这篇文章主要介绍了Redis使用RedisTemplate模板类的常用操作方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-09-09
  • SpringBoot一个非常蛋疼的无法启动的问题解决

    SpringBoot一个非常蛋疼的无法启动的问题解决

    这篇文章主要介绍了SpringBoot一个非常蛋疼的无法启动的问题解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-11-11

最新评论