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

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

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

一、观察者模式##

定义###

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

特点###

  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. Go语言学习索引

    <Go并发编程实战>示例项目 项目地址: https://github.com/hyper-carrot/goc2p 项目安装: 用git clone获取项目,并将其根目录作为一个工作区. ...

  2. ubuntu网桥设置

    什么是桥接? 桥接(Bridging),是指依据OSI网络模型的链路层的地址,对网络数据包进行转发的过程,工作在OSI的第二层.一般的交换机,网桥就有桥接作用. 一般的交换机,网桥就有桥接作用.就交换 ...

  3. ELK日志管理搭建

    目录: 一.介绍 二.安装JDK 三.安装Elasticsearch 四.安装Kibana 五.安装Nginx 六.安装Logstash 七.安装Logstash-forwarder 八.测试 系统环 ...

  4. 程序管理与SElinux

    一.程序: 1.在Linux中,触发任何一个事件是,系统都会将他定义为一个程序,并且给予这个程序一PID,同时依据启发这个程序的使用者与相关属性关系,给予这个PID一组有效的权限设定,从此以后,这个P ...

  5. mongodb查询操作分析

    背景 mongodb 提供了类sql的数据查询及操作方式,同时也包含了聚合操作.索引等多个机制: 按以往的经验,不当的库表操作或索引模式往往会造成许多问题,如查询操作缓慢.数据库吞吐量低下.CPU或磁 ...

  6. Hello 2018

    愈发觉得写技术博客对于自己写作能力的提升会很有帮助,于是在今天终于用Github+Jekyll的方式搭建了自己的博客,从今往后就在这里记录自己在技术上成长的点滴,希望自己的总结和思考也能帮助到其他人. ...

  7. 基于redis的cas实现

    cas是我们常用的一种解决并发问题的手段,小到CPU指令集,大到分布式存储,都能看到cas的影子.本文假定你已经充分理解一般的cas方案,如果你还不知道cas是什么,请自行百度 我们在进行关系型数据库 ...

  8. JMX与Spring

    1.什么是JMX JMX可以监控类的运行情况,可以在程序运行期查看并修改类属性的信息. 举一个应用实例:在一个系统中常常会有一些配置信息,比如服务的IP地址,端口号什么的,那么如何来写这些代码呢? 下 ...

  9. BZOJ 3622: 已经没有什么好害怕的了 [容斥原理 DP]

    3622: 已经没有什么好害怕的了 题意:和我签订契约,成为魔法少女吧 真·题意:零食魔女夏洛特的结界里有糖果a和药片b各n个,两两配对,a>b的配对比b>a的配对多k个学姐就可能获胜,求 ...

  10. IntelliJ IDEA使用心得之基础篇

    今天和大家分享一个非常好用的Java开发工具-IntelliJ IDEA. 下载地址:https://www.jetbrains.com/idea/ 目录: 1)IntelliJ IDEA使用心得之基 ...