龙空技术网

波哥带你探寻SpringBoot中优雅设计监听器的本质

波哥带你学Java 514

前言:

现在咱们对“监听模式 java”大体比较关注,同学们都需要了解一些“监听模式 java”的相关内容。那么小编在网摘上搜集了一些关于“监听模式 java””的相关知识,希望姐妹们能喜欢,你们一起来了解一下吧!

请添加图片描述

SpringBoot源码之监听器设计1.观察者模式

  监听器的设计会使用到Java设计模式中的观察者模式,所以在搞清楚SpringBoot中的监听器的设计之前我们还是非常有必要把观察者模式先弄清楚。

  观察者模式又称为发布/订阅(Publish/Subscribe)模式,在对象之间定义了一对多的依赖,这样一来,当一个对象改变状态,依赖它的对象会收到通知并自动更新.

  在java.util包中包含有基本的Observer接口和Observable抽象类.功能上和Subject接口和Observer接口类似.不过在使用上,就方便多了,因为许多功能比如说注册,删除,通知观察者的那些功能已经内置好了.

1.1 定义具体被观察者

 1package com.dpb.observer2; 2 3import java.util.Observable; 4 5/** 6 * 目标对象 7 * 继承 Observable 8 * @author dengp 9 *10 */11public class ConcreteSubject extends Observable {1213    private int state; 1415    public void set(int s){16        state = s;  //目标对象的状态发生了改变17        setChanged();  //表示目标对象已经做了更改18        notifyObservers(state);  //通知所有的观察者19    }2021    public int getState() {22        return state;23    }2425    public void setState(int state) {26        this.state = state;27    }28}

观察者只需要继承Observable父类。发送消息的方式执行如下两行代码即可

1setChanged();  //表示目标对象已经做了更改2notifyObservers(state);  //通知所有的观察者

Observable源码对应的是:

在这里插入图片描述

在这里插入图片描述

1.2 定义具体观察者

 1package com.dpb.observer2; 2 3import java.util.Observable; 4import java.util.Observer; 5/** 6 * 观察者模式:观察者(消息订阅者) 7 * 实现Observer接口 8 * @author dengp 9 *10 */11public class ObserverA implements Observer {1213    private int myState;1415    @Override16    public void update(Observable o, Object arg) {17        myState = ((ConcreteSubject)o).getState();18    }19    public int getMyState() {20        return myState;21    }22    public void setMyState(int myState) {23        this.myState = myState;24    }25}

观察者也就是订阅者只需要实现Observer接口并重写相关update方法即可,在目标实现中我们发现触发的时候执行的就是观察者的update方法。

1.3 测试

 1package com.dpb.observer2; 2 3public class Client { 4    public static void main(String[] args) { 5        //创建目标对象Obserable 6        ConcreteSubject subject = new ConcreteSubject(); 7 8        //创建观察者 9        ObserverA obs1 = new ObserverA();10        ObserverA obs2 = new ObserverA();11        ObserverA obs3 = new ObserverA();1213        //将上面三个观察者对象添加到目标对象subject的观察者容器中14        subject.addObserver(obs1);15        subject.addObserver(obs2);16        subject.addObserver(obs3);1718        //改变subject对象的状态19        subject.set(3000);20        System.out.println("===============状态修改了!");21        //观察者的状态发生了变化22        System.out.println(obs1.getMyState());23        System.out.println(obs2.getMyState());24        System.out.println(obs3.getMyState());2526        subject.set(600);27        System.out.println("===============状态修改了!");28        //观察者的状态发生了变化29        System.out.println(obs1.getMyState());30        System.out.println(obs2.getMyState());31        System.out.println(obs3.getMyState());3233        //移除一个订阅者34        subject.deleteObserver(obs2);35        subject.set(100);36        System.out.println("===============状态修改了!");37        //观察者的状态发生了变化38        System.out.println(obs1.getMyState());39        System.out.println(obs2.getMyState());40        System.out.println(obs3.getMyState());41    }42}

在这里插入图片描述

  这样就实现了官方提供观察者模式.

2.SpringBoot中监听器的设计

  然后我们来看下SpringBoot启动这涉及到的监听器这块是如何实现的。

2.1 初始化操作

  通过前面的介绍我们知道在SpringApplication的构造方法中会加载所有声明在spring.factories中的监听器。

image.png

  通过Debug模式我们可以看到加载的监听器有哪些。

image.png

  其实就是加载的spring.factories文件中的key为ApplicationListener的value

image.png

image.png

  通过对这些内置监听器的源码查看我们发现这些监听器都实现了 ApplicationEvent接口。也就是都会监听 ApplicationEvent发布的相关的事件。ApplicationContext事件机制是观察者设计模式的实现,通过ApplicationEvent类和ApplicationListener接口,可以实现ApplicationContext事件处理。

image.png

2.2 run方法

  然后我们来看下在SpringApplication.run()方法中是如何发布对应的事件的。

image.png

  首先会通过getRunListeners方法来获取我们在spring.factories中定义的SpringApplicationRunListener类型的实例。也就是EventPublishingRunListener.

image.png

image.png

image.png

  加载这个类型的时候会同步的完成实例化。

image.png

image.png

  实例化操作就会执行EventPublishingRunListener.

image.png

  在这个构造方法中会绑定我们前面加载的11个过滤器。

image.png

  到这其实我们就已经清楚了EventPublishingRunListener和我们前面加载的11个监听器的关系了。然后在看事件发布的方法。

image.png

查看starting()方法。

image.png

再进入

image.png

进入到multicastEvent中方法中我们可以看到具体的触发逻辑

image.png

在这儿以ConfigFileApplicationListener为例。

image.png

触发会进入ConfigFileApplicationListener对象的onApplicationEvent方法中,

image.png

通过代码我们可以发现当前的事件是ApplicationStartingEvent事件,都不满足,所以ConfigFileApplicationListener在SpringBoot项目开始启动的时候就不会做任何的操作。而当我们在配置环境信息的时候,会发布对应的事件来触发

image.png

image.png

继续进入

image.png

继续进入

image.png

然后再触发ConfigFileApplicationListener监听器的时候就会触发如下方法了

image.png

  其实到这儿,后面的事件发布与监听器的处理逻辑就差不多是一致了。到这儿对应SpringBoot中的监听器这块就分析的差不错了。像SpringBoot的属性文件中的信息什么时候加载的就是在这些内置的监听器中完成的。

image.png

官方内置的事件有:

image.png

  好了本文就给大家介绍到这里,希望能对你有所帮助哦。

请添加图片描述

标签: #监听模式 java