认识观察者模式

首先来看看报纸订阅的过程

  • 1.报社的业务就是出版报纸
  • 2.向某家报社订阅报纸,只要他们有新报纸出版,就会送过来,只要你是他们的订户
  • 3.当你不想再看报纸的时候,取消订阅,他们就不会再送新报纸
  • 4.只要报社还在运营,就会一直有人来订阅或取消订阅报纸

观察者模式和报纸订阅流程是一样的,只是名字不同。出版社改名为主题(Subject),而订阅者改名为观察者(Observer)

观察者模式的定义

观察者模式定义了对象之间的一对多依赖,这样一来,当一个对象改变状态时,它的所有依赖者都会受到通知并自动更新。

观察者模式代码

//主题接口
interface Subject {
    void registerObserver(Observer o);//增加观察者
    void removeObserver(Observer o);//删除观察者
    void notifyObservers();//当主题状态改变时,调用此方法通知所有观察者
}
//观察者接口
interface Observer {
    void update(float temp); //用来接收主题发送的通知,这里假设主题是温度计,通过接口给观察者发送当前温度
}
//主题实现
class Thermometer implements Subject {
    private List<Observer> observers = new ArrayList<>();//观察者集合
    private float temp = 0.0f;//温度
    public void setTemp(float temp) {//温度更新
        this.temp = temp;
        notifyObservers();
    }
    @Override
    public void registerObserver(Observer o) {
        observers.add(o);
    }
    @Override
    public void removeObserver(Observer o) {
        if (observers.indexOf(o) >= 0)
            observers.remove(o);
    }
    @Override
    public void notifyObservers() {
        for (Observer observer : observers) {
            observer.update(temp);
        }
    }
}
//观察者实现
class ObserverA implements Observer {
    @Override
    public void update(float temp) {
        System.out.println("观察者A:当前的温度是" + temp);
    }
}
class ObserverB implements Observer {
    @Override
    public void update(float temp) {
        System.out.println("观察者B:当前的温度是" + temp);
    }
}
public class JavaApplication {
    public static void main(String[] args) throws Exception{
       //具体使用
       Thermometer t = new Thermometer();
       Observer a = new ObserverA();
       Observer b = new ObserverB();
       //为a,b注册成为ts的观察者
       t.registerObserver(a);
       t.registerObserver(b);
       //更新温度
       t.setTemp(23.00f);
       //a不再当观察者
       t.removeObserver(a);
       //更新温度
       t.setTemp(24.00f);
    }
}

另外,Java中已经内置了观察者模式,在java.util包下有Observer接口和Observable类

和这里的Subject接口和Observer接口很相似,修改后的代码如下

//主题实现
class Thermometer extends Observable{
    private float temp = 0.0f;//温度
    public void setTemp(float temp) {//温度更新
        this.temp = temp;
        setChanged();//说明状态已改变
        notifyObservers(temp);
    }
}
//观察者实现
class ObserverA implements Observer {
    @Override
    public void update(Observable o, Object arg) {
        System.out.println("观察者A:当前的温度是" + arg);
    }
}
class ObserverB implements Observer {
    @Override
    public void update(Observable o, Object arg) {
        System.out.println("观察者B:当前的温度是" + arg);
    }
}
public class JavaApplication {
    public static void main(String[] args) throws Exception{
       //具体使用
       Thermometer t = new Thermometer();
       Observer a = new ObserverA();
       Observer b = new ObserverB();
       //为a,b注册成为ts的观察者
       t.addObserver(a);
       t.addObserver(b);
       //更新温度
       t.setTemp(23.00f);
       //a不再当观察者
       t.deleteObserver(a);
       //更新温度
       t.setTemp(24.00f);
    }
}

另外,既然JDK是开源的,那么就顺便去看看JDK的源码中的具体实现

(这里使用的源码版本为jdk1.8.0_74)

Observable对象:(大部分没什么区别的代码就不放出来了)

public Class Observable {
    private boolean changed = false;
    private Vector<Observer> obs;

    ...
}

私有成员变量,一个是标志,用于后续判断是否通知观察者,一个观察者集合obs,使用的是Vector容器

标志的用途是用于增加自由度,比如说一秒钟更新10次数据,对于前台显示而言可能不需要这么频繁,那么我们可以自己计数,如果达到10次才设置标志

可以在通知观察者方法源码里看到changed的作用

public void notifyObservers(Object arg) {
    Object[] arrLocal;
    synchronized (this) {
        if (!changed)
            return;
        arrLocal = obs.toArray();
        clearChanged();//clearChanged方法里的具体代码就一句:changed = false;
    }
    for (int i = arrLocal.length-1; i>=0; i--)
        ((Observer)arrLocal[i]).update(this, arg);
}

而Observer接口更是没什么好说的

public interface Observer {
    void update(Observable o, Object arg);
}

相对于我们自己实现的观察者模式,主要区别在于将ArrayList换成了Vector,方法都加上了synchronized关键字来保证线程安全,并没有什么特别的← ←

Head First Design Patterns学习笔记-观察者模式的更多相关文章

  1. how to design Programs 学习笔记

    how to design Programs 学习笔记 */--> how to design Programs 学习笔记 目录 1. 前言 1.1. 系统化程序设计 1.2. 输入和输出 2. ...

  2. 《Head First 设计模式》学习笔记——观察者模式 + 装饰者模式

    装饰者模式是JDK中还有一个使用较多的设计模式,上一个是观察者模式(在Swing中大量使用),业内好的API设计无一离不开常见的设计模式,通常我们所说要阅读源代码,也是为了学习大牛们的设计思路.--- ...

  3. 学习笔记——观察者模式Observer

    观察者模式,当事件发生时,调用相应观察者的方法进行“通知”.Subject中使用一个数据结构存储需要通知的观察者对象,执行Notify时,执行所有观察者的Update方法.

  4. C#学习笔记-观察者模式

    题目1:几个同事为了在上班期间偷偷看休息,做点其他的事情,就和小秘偷偷联系了一下,如果老板回来了,就麻烦小秘偷偷通知一声,这样方便大家及时变更自己的工作状态. 分析: 根据题目分析,首先明确,肯定会有 ...

  5. Java 学习笔记 观察者模式与适配者模式 监控文件夹

    2019.4.12 适配器模式,如果某个类需要实现多个方法,但是开发过程中只需要实现其中的一两个方法,直接使用该类的适配器类,复写需要实现的方法即可 观察者模式:有多个观察者observe观察一个主体 ...

  6. Fusion360_Generative Design 入门学习笔记

    2019.12.17更新 初次见到衍生式设计的时候感觉非常惊艳,现在觉得这个功能就是个弟弟,只能做一些中看不中用的东西.这个方法的理论基础是拓扑优化,想做research的同学可参阅"如何入 ...

  7. Ant Design Pro 学习笔记:数据流向

    在讲这个问题之前,有一个问题应当讲一下: Ant Design Pro / umi / dva 是什么关系? 首先是 umi / dva 的关系. umi 是一个基于路由的 react 开发框架. d ...

  8. Java-马士兵设计模式学习笔记-观察者模式-模拟Awt Button

    一.概述 Java 的Awt是 Observer模式,现用Java自己模拟awt中Button的运行机制 二.代码 1.Test.java import java.text.DateFormat; i ...

  9. Java-马士兵设计模式学习笔记-观察者模式-AWT简单例子

    1.AWT简单例子 TestFrame.java import java.awt.Button; import java.awt.Frame; import java.awt.event.Action ...

随机推荐

  1. javaScript数据类型与typeof操作符

    1,typeof操作符. typeof操作符是用来检测变量的数据类型.使用:typeof  变量名;返回以下字符串: 字符串 描述 undefined 未定义 boolean 布尔值 string 字 ...

  2. js深拷贝和浅拷贝

    一.数组的深浅拷贝 在使用JavaScript对数组进行操作的时候,我们经常需要将数组进行备份,事实证明如果只是简单的将它赋予其他变量,那么我们只要更改其中的任何一个,然后其他的也会跟着改变,这就导致 ...

  3. Chart 点击获取坐标

    private void chart2_MouseMove(object sender, MouseEventArgs e) { if (!this.DesignMode) { ].AxisX.Sca ...

  4. asp.net 客户端上传文件全路径获取方法

    asp.net  获取客户端上传文件全路径方法: eg:F:\test\1.doc 基于浏览器安全问题,浏览器将屏蔽获取客户端文件全路径的方法,只能获取到文件的文件名,如果需要获取全路径则需要另想其他 ...

  5. JS文档和Demo自动化生成工具 - SmartDoc发布

    曾几何时,当你码神附体,一路披荆斩棘的完成代码后,带着“一码在手,天下我有”的傲然环顾之时,却发现单元测试.API文档.Demo实例陆续向你砸来,顿时有木有一种冰水挑战后的感觉.而这时你应该:哟哟,快 ...

  6. Portal for ArcGIS上传shp文件中文乱码可能情况

    环境: windows Server 2008:Portal for ArcGIS 10.2:ArcGIS for Server 10.2:PostgreSQL 9.2:ArcGIS for Desk ...

  7. CentOS6部署VNC服务端

    VNC (Virtual Network Computer)是虚拟网络计算机的缩写.VNC 是在基于 UNIX 和 Linux 操作系统的免费的开源软件,远程控制能力强大,高效实用,其性能可以和 Wi ...

  8. 随着ScrollView的滑动,渐渐的执行动画View

    今天是实现了一个小功能的东西.看看效果图: 实现方式: 1.自定义ScrollView   复写onScrollChange方法,来计算滑动的位置. 2.自定义接口,通过接口来在ScrollView中 ...

  9. Android性能优化之内存篇

    下面是内存篇章的学习笔记,部分内容与前面的性能优化典范有重合,欢迎大家一起学习交流! 1)Memory, GC, and Performance 众所周知,与C/C++需要通过手动编码来申请以及释放内 ...

  10. Keepalived 使用指南

    Keepalived 使用指南 1.    简介 负载均衡是虚拟服务的一种好的处理方案.当设计一种负载均衡的拓扑时一定要考虑到如下两点: 真实服务器的可用性使用健康检测机制. 负载均衡器的可用性使用故 ...