Java中23种面试常考的设计模式之观察者模式(Observer)---行为型模式

简介: Java中23种面试常考的设计模式之观察者模式(Observer)---行为型模式

【Java中23种面试常考的设计模式之观察者模式(Observer)---行为型模式】

知识回顾:
>

之前我们讲过的设计模式在这里呦:
【面试最常见的设计模式之单例模式】
【面试最常见的设计模式之工厂模式】
【Java中23种面试常考的设计模式之备忘录模式(Memento)---行为型模式】
接下来我们要进行学习的是:【Java中23种面试常考的设计模式之观察者模式(Observer)---行为型模式】。

观察者模式

观察者模式又称为发布/订阅(Publish/Subscribe)模式,当对象间存在一对多关系时,则使用观察者模式(Observer Pattern)。比如,当一个对象被修改时,则会自动通知依赖它的对象。观察者模式属于行为型模式。

生产生活中常用的使用场景

  1. 微信公众号订阅;
  2. Redis中订阅与发布;
  3. Kafka消息中间件中生产者发送消息给消费者进行消费;
  4. 某个店中注册了会员,一有消息就会通知所有的会员;
  5. 监听器实现;
  6. 广播器的实现;
  7. 等等。。。。。等等。。。。

观察者模式优点与缺点

优点

  1. 一个对象状态改变给其他关注这个对象进行通知的。

缺点

  1. 如果一个被观察者对象有很多的直接和间接的观察者的话,将所有的观察者都通知到会花费很多时间。
  2. 要避免发生循环依赖,否则会导致系统的奔溃。

核心内容

观察者模式中最重要的是 Subject(接口以及具体实现类)、Observer(接口以及具体实现类) 。
抽象被观察者角色:也就是一个抽象主题,它把所有对观察者对象的引用保存在一个集合中,每个主题都可以有任意数量的观察者。抽象主题提供一个接口,可以增加和删除观察者角色。
抽象观察者角色:为所有的具体观察者定义一个接口。
具体被观察者角色:是一个具体的主题,所有登记过的观察者发出通知。
具体观察者角色:实现抽象观察者角色所需要的更新接口。

UML类图

image.png

实现代码

抽象主题(被观察者)

package com.observer;

import java.util.ArrayList;
import java.util.List;

public interface Subject<T> {
   
   
    // 通过集合来装载订阅者
    List<Observer> list = new ArrayList<>();
    // 添加订阅者
    public void addObserver(T t);
    // 删除订阅者
    public void deleteObserver(T t);
    //通知所有的观察者更新状态
    public void notifyAllObservers();
}

抽象观察者

package com.observer;

public interface Observer {
   
   

    void  change(Subject subject);
}

具体主题(具体被观察者)

package com.observer;

public class Topic1 implements Subject<Observer> {
   
   

    //此时具体主题的状态
    private int topicState;

    public int getTopicState() {
   
   
        return topicState;
    }

    public void setTopicState(int topicState) {
   
   
        this.topicState= topicState;
        //主题对象值发生了变化,请通知所有的观察者
        this.notifyAllObservers();
    }

    // 更新每一个观察者中的信息
    @Override
    public void notifyAllObservers() {
   
   
        for (Observer obs : list) {
   
   
            obs.change(this);
        }
    }

    // 添加指定的观察者
    @Override
    public void addObserver(Observer obs) {
   
   
        list.add(obs);
    }

    // 删除指定的观察者
    @Override
    public void deleteObserver(Observer obs) {
   
   
        list.remove(obs);
    }
}

具体观察者

package com.observer;


public class Observer1 implements Observer {
   
   

    //observerState需要跟目标对象的topicState值保持一致
    private int observerState;   

    public int getObserverState() {
   
   
        return observerState;
    }

    public void setObserverState(int observerState) {
   
   
        this.observerState= observerState;
    }

    //此时主题的状态已经更改了,Observer1更改为次时主题更改后的状态
    @Override
    public void change(Subject subject) {
   
   
        observerState = ((Topic1)subject).getTopicState();
    }

}

测试代码

package com.observer;

public class Main{
   
   
    public static void main(String[] args) {
   
   
        //主题
        Topic1 subject = new Topic1();

        //创建多个观察者
        Observer1 obs1 = new Observer1();
        Observer1 obs2 = new Observer1();
        Observer1 obs3 = new Observer1();

        //将这三个观察者添加到subject对象的观察者队伍中
        subject.addObserver(obs1);
        subject.addObserver(obs2);
        subject.addObserver(obs3);


        //举个栗子1:先设置subject的状态为1314
        System.out.println("---------------先举一个栗子-------------------");
        subject.setTopicState(1314);
        System.out.println("此时subject主题中的状态是:"+subject.getTopicState());
        System.out.println("接下来我们看一下观察者们的状态是否发生了改变:");
        System.out.println(obs1.getObserverState());
        System.out.println(obs2.getObserverState());
        System.out.println(obs3.getObserverState());
        System.out.println("---------------再来一个栗子-------------------");
        //举个栗子2:先设置subject的状态为521
        subject.setTopicState(521);
        System.out.println("此时subject主题中的状态是:"+subject.getTopicState());
        System.out.println("接下来我们看一下观察者们的状态是否发生了改变:");
        System.out.println(obs1.getObserverState());
        System.out.println(obs2.getObserverState());
        System.out.println(obs3.getObserverState());

    }
}

运行结果展示:

image.png

补充:Java内置观察者模式实现

  在java.util包中包含有基本的Observer接口和Observable抽象类.一些方法都帮我们封装好了,我们的代码就参考这里,这里就不再做详细的赘述了,感兴趣的同学自己可以去实现一下。

好了,到这里【Java中23种面试常考的设计模式之观察者模式(Observer)---行为型模式】就结束了,23中设计模式持续更新汇总中。

相关文章
|
2天前
|
Java
Java面向对象实践小结(含面试题)(下)
Java面向对象实践小结(含面试题)(下)
11 1
|
3天前
|
设计模式 安全 Java
【JAVA】Java 中什么叫单例设计模式?请用 Java 写出线程安全的单例模式
【JAVA】Java 中什么叫单例设计模式?请用 Java 写出线程安全的单例模式
|
2天前
|
设计模式 Java
Java面向对象实践小结(含面试题)(上)
Java面向对象实践小结(含面试题)
12 1
|
3天前
|
设计模式 消息中间件 Java
Java 设计模式:探索发布-订阅模式的原理与应用
【4月更文挑战第27天】发布-订阅模式是一种消息传递范式,被广泛用于构建松散耦合的系统。在 Java 中,这种模式允许多个对象监听和响应感兴趣的事件。
20 2
|
3天前
|
设计模式 算法 Java
Java 设计模式:探索策略模式的概念和实战应用
【4月更文挑战第27天】策略模式是一种行为设计模式,它允许在运行时选择算法的行为。在 Java 中,策略模式通过定义一系列的算法,并将每一个算法封装起来,并使它们可以互换,这样算法的变化不会影响到使用算法的客户。
10 1
|
3天前
|
设计模式 Java
Java 设计模式:工厂模式与抽象工厂模式的解析与应用
【4月更文挑战第27天】设计模式是软件开发中用于解决常见问题的典型解决方案。在 Java 中,工厂模式和抽象工厂模式是创建型模式中非常核心的模式,它们主要用于对象的创建,有助于增加程序的灵活性和扩展性。本博客将详细介绍这两种模式的概念、区别以及如何在实际项目中应用这些模式。
8 1
|
4天前
|
JavaScript 前端开发 Java
【JAVA面试题】什么是引用传递?什么是值传递?
【JAVA面试题】什么是引用传递?什么是值传递?
|
6天前
|
存储 安全 Java
[Java基础面试题] Map 接口相关
[Java基础面试题] Map 接口相关
|
6天前
|
Java
[Java 面试题] ArrayList篇
[Java 面试题] ArrayList篇
|
6天前
|
设计模式 算法 Java
[设计模式Java实现附plantuml源码~行为型]定义算法的框架——模板方法模式
[设计模式Java实现附plantuml源码~行为型]定义算法的框架——模板方法模式