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

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

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

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

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

代码

抽象主题

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. 顶部选项卡-可左右拖动(webview)示例如何做到tab与webview联动滚动

    顶部选项卡-可左右拖动(webview)的示例中,如何做到tab与webview联动滚动,效果类似uc头条一样 ps:自己也不确定有多少了到航头,页面怎么办,到航头从后台获取,页面不可能建N多个.ht ...

  2. scala 学习笔记--闭了个包

    object Test01 { def main(args: Array[String]): Unit = { //普通函数 val squ=(x:Int) => x*x // ps:_的简写形 ...

  3. micronaut 学习 二 创建一个简单的服务

    micronaut 提供的cli 很方便,我们可以快速创建具有所需特性的应用,以下是一个简单的web server app 创建命令 mn create-app hello-world 效果 mn c ...

  4. ABP 04 用户的创建

    有这样一个问题,我忘记了密码,查了一下数据那张表,是加密了的,然后就有了这篇文章了. 往后台传的时候,还是传的明文. 请求的地址:/api/services/app/User/Create 用户还是挺 ...

  5. [Java] key

    Z2VueW1vJTIwJTI2JTI2JTI2JTIwMTYzJTNBJTBBdXNyJTIwLSUyMHd1a29uZ3N1bjEyMzQlMjAlMjMlMjAxNjMuc3VmZml4JTIw ...

  6. nginx之动静分离(nginx与php不在同一台服务器)

    nginx实现动静分离(nginx与php不在同一个服务器) 使用wordpress-5.0.3-zh_CN.tar.gz做实验 Nginx服务器的配置: [root@app ~]# tar xf w ...

  7. X-factor Chain(信息学奥赛一本通 1628)

    题目描述 输入正整数 x,求 x 的大于 1 的因子组成的满足任意前一项都能整除后一项的序列的最大长度,以及满足最大长度的序列的个数. 输入 多组数据,每组数据一行,包含一个正整数 x. 对于全部数据 ...

  8. Linux后台运行和关闭程序、查看后台任务

    fg.bg.jobs.&.ctrl+z   1.&    (最经常被用到)     这个用在一个命令的最后,可以把这个命令放到后台执行   2.ctrl + z     可以将一个正在 ...

  9. SpringCloud-Feign声明式服务调用

    在前面的文章中可以发现当我们通过RestTemplate调用其它服务的API时,所需要的参数须在请求的URL中进行拼接,如果参数少的话或许我们还可以忍受,一旦有多个参数的话,这时拼接请求字符串就会效率 ...

  10. vue.js动态绑定input的checked

    不管<input type='radio checked='true''>  你的checked属性值是true或者false,他都会选中. 其实原理是这样的,复选框里只要有checked ...