相信大家都有看过《喜洋洋与灰太狼》,说的是灰太狼和羊族的“斗争”,而每次的结果都是灰太狼一飞冲天,伴随着一句“我还会回来的......”。为灰太狼感到悲哀,抓不到羊,在家也被老婆平底锅虐待。灰太狼为什么会这么背?

很简单,灰太狼本身就有“暴露行踪”的属性,羊咩咩就能知晓灰太狼要干嘛,不背才怪呢。

为了帮助灰太狼摆脱被老婆平底锅抽的悲剧,发起了“解救灰太狼”的行动,必须要知道观察者模式

一、观察者模式##

定义###

  观察者模式又叫做发布-订阅模式,定义了对象间一对多的依赖关系,使得当对象状态发生变化时,所有依赖它的对象都会收到通知并且自动更新自己。

特点###

  1)被观察者需要持有一个或者多个观察者对象。

  2)系统中一个模块的变化,某些模块也会跟随着变化。

UML###

从上面的UML可以看出来,观察者模式设计到的角色有如下四个:

  - 抽象被观察者角色:定义了动态增加、删除以及通知观察者对象的方法,职责就是管理和通知观察者。持有观察者对象的集合。

  - 具体被观察者角色:一般继承抽象被观察者,实现自己本身的业务逻辑,当状态发生改变时发起通知。

  - 抽象观察者角色:提供一个接口,定义了观察者收到通知时更新自己的方法。

  - 具体观察者角色:实现抽象观察者接口,处理不同具体观察者的不同业务逻辑。

二、实战##

灰太狼具有被观察者属性,喜洋洋这些羊咩咩一直都在观察者灰太狼,所以羊咩咩们是观察者。OK,角色确定了,看看具体是怎么实现的...

抽象被观察者代码如下:

public abstract class Subject {

    /**
* 观察者对象的集合
*/
private List<Observer> observerList = new ArrayList<>(); /**
* 登记观察者
*
* @param observer
*/
public void attach(Observer observer) {
observerList.add(observer);
System.out.println("增加了观察者:" + observer.getName());
} /**
* 删除观察者
*
* @param observer
*/
public void dettach(Observer observer) {
observerList.remove(observer);
System.out.println("删除了观察者:" + observer.getName());
} /**
* 通知所有观察者
*/
public void notifyObserver() {
for (Observer observer : observerList) {
observer.update("灰太狼要搞事情了");
}
} }

灰太狼是具体被观察者,继承抽象被观察者,代码如下:

public class Wolf extends Subject {

    public void invade(){

        System.out.println("灰太狼:我要搞事情了");
// 通知所有观察者
notifyObserver();
} }

抽象观察者代码如下:

public interface Observer {

    String getName();

    /**
* 通知更新方法
*
* @param msg
*/
public void update(String msg); }

喜羊羊是具体观察者,实现抽象观察者,代码如下:

public class PleasantSheep implements Observer{

    @Override
public String getName() {
return "喜羊羊";
} /**
* 具体业务逻辑
*/
@Override
public void update(String msg) {
System.out.println("喜羊羊收到通知:" + msg);
} }

接下来看客户端如何把观察者模式跑起来,代码如下:

public class Client {

    public static void main(String[] args) {
// 灰太狼--被观察者
Wolf wolf = new Wolf();
// 喜羊羊--观察者
Observer pleasantSheep = new PleasantSheep();
// 登记观察者
wolf.attach(pleasantSheep);
// 灰太狼入侵
wolf.invade();
} }

运行客户端代码,结果如下:

增加了观察者:喜羊羊

灰太狼:我要搞事情了

喜羊羊收到通知:灰太狼要搞事情了

看到了吧,灰太狼这不是自找虐吗!搞事情还要发通知,活该被平底锅拍飞。灰太狼不止通知了喜羊羊,还通知了懒羊羊。

懒羊羊也是具体观察者,代码如下:

public class LazySheep implements Observer {

    @Override
public String getName() {
return "懒羊羊";
} @Override
public void update(String msg) {
System.out.println("懒羊羊收到通知:" + msg);
} }

客户端代码如下:

public class Client {

    public static void main(String[] args) {
// 灰太狼--被观察者
Wolf wolf = new Wolf(); // 喜羊羊--观察者
Observer pleasantSheep = new PleasantSheep();
// 登记观察者
wolf.attach(pleasantSheep); // 懒羊羊--观察者
Observer lazySheep = new LazySheep();
// 登记观察者
wolf.attach(lazySheep); // 灰太狼入侵
wolf.invade();
} }

上面客户端代码创建了一个懒羊羊观察者,添加了观察者集合中,这样懒羊羊也会受到通知,运行结果如下:

增加了观察者:喜羊羊

增加了观察者:懒羊羊

灰太狼:我要搞事情了

喜羊羊收到通知:灰太狼要搞事情了

懒羊羊收到通知:灰太狼要搞事情了

那如何帮助灰太狼摆脱这个命运呢,把观察者从集合中移除就OK了,代码如下:

public class Client {

    public static void main(String[] args) {
// 灰太狼--被观察者
Wolf wolf = new Wolf(); // 喜羊羊--观察者
Observer pleasantSheep = new PleasantSheep();
// 登记观察者
wolf.attach(pleasantSheep); // 懒羊羊--观察者
Observer lazySheep = new LazySheep();
// 登记观察者
wolf.attach(lazySheep); // 灰太狼入侵
wolf.invade(); // 删除观察者
wolf.dettach(pleasantSheep); wolf.invade();
} }

再次运行客户端,结果如下:

增加了观察者:喜羊羊

增加了观察者:懒羊羊

灰太狼:我要搞事情了

喜羊羊收到通知:灰太狼要搞事情了

懒羊羊收到通知:灰太狼要搞事情了

删除了观察者:喜羊羊

灰太狼:我要搞事情了

懒羊羊收到通知:灰太狼要搞事情了

可以看到,把喜羊羊从观察者集合中移除了,它就不会再收到通知。

三、观察者模式的优缺点##

优点###

1)观察者和被观察者之间抽象耦合。观察者模式容易扩展,被观察者只持有观察者集合,并不需要知道具体观察者内部的实现。

2)对象之间的保持高度的协作。当被观察者发生变化时,所有被观察者都会通知到,然后做出相应的动作。

缺点

1)如果观察者太多,被观察者通知观察者消耗的时间很多,影响系统的性能。

2)当观察者集合中的某一观察者错误时就会导致系统卡壳,因此一般会采用异步方式。

四、比较##

跟代理模式对比:观察者模式和代理模式主要区别在它们功能不一样,观察者模式强调的是被观察者反馈结果,而代理模式是同根负责做同样的事情。

总结##

在Java中已经提供了Observable类以及一个Observer接口,也就是说Java已经实现了观察者模式的定义,可看出观察者模式在程序系统中的使用率是很高的,不单是Java,Android中也经常看到观察者模式的运用,比如OnClickListener,Rxjava等。下一篇会补上属于创建型模式的原型模式,下回分解,再见。

设计模式Java源码GitHub下载https://github.com/jetLee92/DesignPattern

我的Java设计模式-观察者模式的更多相关文章

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

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

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

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

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

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

  4. Java设计模式--观察者模式到监听器

    观察者模式是对象的行为模式.又叫做发布-订阅模式.模型-视图模式.源-监听器模式. 抽象主题角色:主题角色将所有对观察者对象的引用到保存在一个集合里,每个主题都可以拥有任意数量的观察者.抽象主题提供一 ...

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

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

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

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

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

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

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

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

  9. JAVA设计模式—观察者模式和Reactor反应堆模式

    被观察者(主题)接口 定义主题对象接口 /**抽象主题角色: 这个主题对象在状态上发生变化时,会通知所有观察者对象 也叫事件对象 */ public interface Subject { //增加一 ...

随机推荐

  1. 将excel文件内容存储到数据库,并可以实时在前端查看(不必生成文件)

    版权声明:本文为博主原创文章,未经博主允许不得转载 本文主要讲前端内容,后端涉及较少,可以认为是使用Java. 首先是excel文件上传,这个较为简单,可以html5的数据接口FormData()进行 ...

  2. tensorflow Image 解码函数

    觉得有用的话,欢迎一起讨论相互学习~Follow Me tf.image.decode_png(contents, channels=None, name=None) Decode a PNG-enc ...

  3. 基于JAVA实现的排序算法总结

    常用的排序方法有:冒泡排序.快速排序.选择排序.插入排序.归并排序,除此之外,还有基数排序.鸡尾酒排序.桶排序.鸽巢排序.希尔排序等,这里着重介绍下前半段列举的几种常见方法的实现. 1. 冒泡排序法: ...

  4. 一道python面试题引发的血案

    这里说的是一道阿里校招的面试题:一行代码实现对列表a中的偶数位置的元素进行加3后求和? 今天去面试同样遇到了这个题目,这道题考察的是对python高阶函数map/filter的灵活运用(具体的使用方法 ...

  5. python3,进程间的通信

    本文来源于python 3.5版本的官方文档 multiprocessing模块为进程间通信提供了两种方法: 1.进程队列queue The Queue class is a near clone o ...

  6. 不使用Math.random实现随机数

    不使用Math.random实现随机数 var rand = (function(){ var today = new Date(); var seed = today.getTime(); func ...

  7. 如何更改图片的背景色(PS、证件照之星)

    如何更改图片的背景色(PS.证件照之星) 1.1  证照之星教你如何给证件照换背景 证照之星教你如何给证件照换背景?这个问题困扰很多人,如果你不了解证照之星,一款专业的证件照片制作软件,你肯定就无法自 ...

  8. UOJ #207. 共价大爷游长沙 [lct 异或]

    #207. 共价大爷游长沙 题意:一棵树,支持加边删边,加入点对,删除点对,询问所有点对是否经过一条边 一开始一直想在边权上做文章,或者从连通分量角度考虑,比较接近正解了,但是没想到给点对分配权值所以 ...

  9. 【转】 C/C++程序员必须熟练应用的开源项目

    作为一个经验丰富的C/C++程序员, 肯定亲手写过各种功能的代码, 比如封装过数据库访问的类, 封装过网络通信的类,封装过日志操作的类, 封装过文件访问的类, 封装过UI界面库等, 也在实际的项目中应 ...

  10. 2018/2/5 ELK技术栈之ElasticSearch学习笔记

    npm config set registry https://registry.npm.taobao.org npm config get registry 支持跨域访问http.cors.enab ...