观察者模式是对象的行为模式。又叫做发布-订阅模式、模型-视图模式、源-监听器模式。

抽象主题角色:主题角色将所有对观察者对象的引用到保存在一个集合里,每个主题都可以拥有任意数量的观察者。抽象主题提供一个接口,可以增加或者删除观察者对象。主题角色又叫被观察者。

具体主题角色:将有关状态存入具体观察者对象,在具体主题的内部状态改变时,给所有登记过的观察者发出通知。具体主题角色是抽象主题的一个具体子类实现。

抽象观察者角色:为所有的具体观察者定义一个接口,在得到主题通知的时候更新自己。

具体观察者角色:抽象观察者的具体子类实现。

代码

抽象主题

package com.demo.exercise.observer;

/**
* 抽象主题
*/
public interface Subject { void add(Observer observer); void del(Observer observer); void notifyObservers(); }

抽象观察者

package com.demo.exercise.observer;

/**
* 抽象观察者
*/
public interface Observer { void update(); }

具体观察者

package com.demo.exercise.observer;

/**
* 具体观察者
*/
public class ConcreteObserver implements Observer {
@Override
public void update() {
System.out.println(this + ":接收到通知了...");
}
}

具体主题

package com.demo.exercise.observer;

import java.util.Iterator;
import java.util.Vector; /**
* 具体主题
*/
public class ConcreteSubject implements Subject { private Vector<Observer> observers = new Vector<>(); @Override
public void add(Observer observer) {
observers.add(observer);
} @Override
public void del(Observer observer) {
observers.remove(observer);
} @Override
public void notifyObservers() {
System.out.println("通知所有观察者...");
Iterator<Observer> iterator = observers.iterator();
while (iterator.hasNext()){
iterator.next().update();
}
}
}

运行

    public static void main(String[] args) {
Observer a = new ConcreteObserver();
Observer b = new ConcreteObserver();
Subject subject = new ConcreteSubject();
subject.add(a);
subject.add(b);
subject.notifyObservers();
}

输出

优化方案

上面的【抽象主题角色】,其中管理集合的方法,都是由子类来实现的,而实际情况则是这些管理集合的方法是所有实现子类共用的,所以可以把这些转移到抽象主题中去。

抽象主题(这里还是用的interface,你也可以用abstract class)

package com.demo.exercise.observer;

import java.util.Iterator;
import java.util.Vector; /**
* 抽象主题
*/
public interface Subject { Vector<Observer> observers = new Vector<>(); default void add(Observer observer){
observers.add(observer);
} default void del(Observer observer){
observers.remove(observer);
} default void notifyObservers(){
System.out.println("通知所有观察者...");
Iterator<Observer> iterator = observers.iterator();
while (iterator.hasNext()){
iterator.next().update();
}
} /**
* 子类需要自定义的方法
* @param status
*/
void change(String status);
}

具体子类

package com.demo.exercise.observer;
/**
* 具体主题
*/
public class ConcreteSubject implements Subject { private String status; /**
* 主题状态改变,调用通知方法
* @param status
*/
public void change(String status){
System.out.println(status);
this.status = status;
this.notifyObservers();
}
}

运行

    public static void main(String[] args) {
Observer a = new ConcreteObserver();
Observer b = new ConcreteObserver();
Subject subject = new ConcreteSubject();
subject.add(a);
subject.add(b);
subject.change("主题状态改变");
}

输出

引申

那么这种设计模式用来解决什么问题呢?类似于下面这种代码,大家初学Java的时候肯定用过,点击按钮事件,只要点击按钮,就会自动执行相关方法。

JButton button = new JButton();
button.addActionListener((event) -> {
// TODO
});

其中的原理也是观察者模式。下面我就来简单模拟一下

抽象主题

package com.demo.exercise.observer;

import java.util.Iterator;
import java.util.Vector; /**
* 抽象主题
*/
public abstract class Button { private Vector<ActionListener> observers = new Vector<>(); void addActionListener(ActionListener observer){
observers.add(observer);
} void removeActionListener(ActionListener observer){
observers.remove(observer);
} void notifyObservers(){
Iterator<ActionListener> iterator = observers.iterator();
while (iterator.hasNext()){
iterator.next().actionPerformed();
}
}
}

抽象观察者

package com.demo.exercise.observer;

/**
* 抽象观察者
*/
public interface ActionListener { void actionPerformed(); }

具体主题

package com.demo.exercise.observer;

/**
* 具体主题
*/
public class JButton extends Button { /**
* 点击事件
*/
public void click(){
System.out.println("【假装产生了点击事件】");
this.notifyObservers();
} }

运行

    public static void main(String[] args) {
JButton button = new JButton();
button.addActionListener(() -> {
System.out.println("点击事件处理...");
});
button.addActionListener(() -> {
System.out.println("另外一个点击事件处理...");
});
button.click();
}

输出

Java对观察者模式的支持

一个是被观察者:java.util.Observable

一个是观察者:java.util.Observer

Observer里面只有一个update方法。

public interface Observer {
/**
* This method is called whenever the observed object is changed. An
* application calls an <tt>Observable</tt> object's
* <code>notifyObservers</code> method to have all the object's
* observers notified of the change.
*
* @param o the observable object.
* @param arg an argument passed to the <code>notifyObservers</code>
* method.
*/
void update(Observable o, Object arg);
}

当被观察者的状态发生改变,就会调用这一方法。

    public void notifyObservers(Object arg) {
/*
* a temporary array buffer, used as a snapshot of the state of
* current Observers.
*/
Object[] arrLocal; synchronized (this) {
/* We don't want the Observer doing callbacks into
* arbitrary code while holding its own Monitor.
* The code where we extract each Observable from
* the Vector and store the state of the Observer
* needs synchronization, but notifying observers
* does not (should not). The worst result of any
* potential race-condition here is that:
* 1) a newly-added Observer will miss a
* notification in progress
* 2) a recently unregistered Observer will be
* wrongly notified when it doesn't care
*/
if (!changed)
return;
arrLocal = obs.toArray();
clearChanged();
} for (int i = arrLocal.length-1; i>=0; i--)
((Observer)arrLocal[i]).update(this, arg);
}

用法呢,我再给你示范一下。

创建一个事件对象

package com.demo.exercise.observer;

/**
* 事件对象
*/
public class ClickEvent { private String source; public ClickEvent(String source) {
this.source = source;
} public String getSource() {
return source;
} public void setSource(String source) {
this.source = source;
}
}
package com.demo.exercise.observer;

import java.util.Observable;
import java.util.Observer; /**
* 具体观察者
*/
public class JJButton extends Observable { public void click(){
// 事件对象
ClickEvent clickEvent = new ClickEvent("点击事件");
// changed 置为true,表示状态改变,才会通知所有观察者,与之对应的是clearChanged
this.setChanged();
// 通知所有观察者
this.notifyObservers(clickEvent);
} public void addListener(Observer o){
this.addObserver(o);
} }

运行

    public static void main(String[] args) {
JJButton button = new JJButton();
button.addListener((o, arg) -> {
ClickEvent event = (ClickEvent) arg;
System.out.println("事件源:" + event.getSource());
});
button.click();
}

输出

Java设计模式--观察者模式到监听器的更多相关文章

  1. java设计模式--观察者模式(Observer)

    java设计模式--观察者模式(Observer) java设计模式--观察者模式(Observer) 观察者模式的定义: 定义对象间的一种一对多的依赖关系.当一个对象的状态发生改变时,所有依赖于它的 ...

  2. 【设计模式】Java设计模式 - 观察者模式

    [设计模式]Java设计模式 - 观察者模式 不断学习才是王道 继续踏上学习之路,学之分享笔记 总有一天我也能像各位大佬一样 @一个有梦有戏的人 @怒放吧德德 分享学习心得,欢迎指正,大家一起学习成长 ...

  3. JAVA 设计模式 观察者模式

    用途 观察者模式 (Observer) 观察者模式定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象. 这个主题对象在状态发生变化时,会通知所有观察者对象,使它们能够自动更新自己. 观 ...

  4. Java设计模式の观察者模式(推拉模型)

    目录: 一.观察者定义 二.观察者模式的结构(推模式实现) 三.推模型和拉模型(拉模式实现) 四.JAVA提供的对观察者模式的支持 五.使用JAVA对观察者模式的支持(自带推模式实现实例) 一.观察者 ...

  5. java设计模式--观察者模式和事件监听器模式

    观察者模式 观察者模式又称为订阅—发布模式,在此模式中,一个目标对象管理所有相依于它的观察者对象,并且在它本身的状态改变时主动发出通知.这通常透过呼叫各观察者所提供的方法来实现.此种模式通常被用来事件 ...

  6. java设计模式-观察者模式学习

    最近学习了设计模式中的观察者模式,在这里记录下学习成果. 观察者模式,个人理解:就是一个一对多模型,一个主体做了事情,其余多个主体都可以观察到.只不过这个主体可以决定谁去观察他,以及做什么事情可以给别 ...

  7. Java设计模式——观察者模式(事件监听)

    最近在看Tomcat和Spring的源码,在启动的时候注册了各种Listener,事件触发的时候就执行,这里就用到了设计模式中的观察者模式. 引-GUI中的事件监听 想想以前在学Java的GUI编程的 ...

  8. Java设计模式 - 观察者模式

    定义 观察者模式属于对象行为型模式. 在对象之间定义一对多的依赖,这样一来当一个对象改变状态,依赖它的对象都会收到通知并自动更新. 优点 1.  主题和观察者之间抽象耦合.无论什么对象主要实现了特定的 ...

  9. 我的Java设计模式-观察者模式

    相信大家都有看过<喜洋洋与灰太狼>,说的是灰太狼和羊族的"斗争",而每次的结果都是灰太狼一飞冲天,伴随着一句"我还会回来的......".为灰太狼感 ...

随机推荐

  1. HDU 6595 Everything Is Generated In Equal Probability (期望dp,线性推导)

    Everything Is Generated In Equal Probability \[ Time Limit: 1000 ms\quad Memory Limit: 131072 kB \] ...

  2. Python -- 正则表达式 regular expression

    正则表达式(regular expression) 根据其英文翻译,re模块 作用:用来匹配字符串.  在Python中,正则表达式是特殊的字符序列,检查一个字符串是否与某种模式匹配. 设计思想:用一 ...

  3. swap file "*.swp" already exists!的解决方法

    Linux下编程难免要开启多个vim共同编辑同一个文件,这时再次保存就会出现: swap file "*.swp" already exists! [O]pen Read-Only ...

  4. shell for 循环演示

    test.sh #!/bin/bash for skill in Ada Coffe Action Java; do echo "I am good at ${skill}Script&qu ...

  5. 第10组 团队Git现场编程实战

    组员职责分工 姓名 分工 童景霖 博客 朱晓倩 制作UI 万本琳 制作UI 唐怡 制作UI 陈心怡 制作UI 黄永福 测评福州最受欢迎的商圈.后期代码修改和完善 郑志强 测评各个价位的前五美食餐厅代码 ...

  6. K8S集群搭建——基于CentOS 7系统

    环境准备集群数量此次使用3台CentOS 7系列机器,分别为7.3,7.4,7.5 节点名称 节点IPmaster 192.168.0.100node1 192.168.0.101node2 192. ...

  7. NSGA-II算法学习

    什么是支配: 支配就是统治,在各方面都优于其余个体 如个体i支配个体j,就说明个体i在所有目标函数的表现上都不差于个体j,并且至少在一个目标上优于个体j: 什么是非支配: 非支配就是个体在种群中是最优 ...

  8. jquery实现select数据回显

    [html] view plain copy     <select class="div_select_re" id="edit_technicalGrade&q ...

  9. Invalid bound statement (not found) 终极解决办法

    网上已经有很多文章说明可能导致这个报错的原因,无非是以下几种:1.检查xml文件的namespace是否正确 2.Mapper.java的方法在Mapper.xml中没有,然后执行Mapper的方法会 ...

  10. vue中axios使用二:axios以post,get,jsonp的方式请求后台数据

    本文为博主原创,转载请注明出处 axios在上一篇中讲过:vue中axios使用一:axios做拦截器,axios是请求后台资源的模块,用来请求后台资源. axios本身是支持get,post请求后台 ...