1、初始化器ApplicationContextInitializer
我們在啟動Spring Boot項目的時候,是執(zhí)行這樣一個方法來啟動的
我們一層一層往下點,最終發(fā)現執(zhí)行的是這個方法
所以我們在啟動項目的時候也可以這樣啟動 new SpringApplication(SpringbootExtensionPointApplication.class).run(args); 老的只是包裝了一個靜態(tài)方法,實際底層就是實例化一個SpringApplication對象,然后調用它的run方法。
我們進到構造方法里看下,紅框里面標出來的,一個是設置初始化器,一個是設置監(jiān)聽器。
點進去看下,這兩個底層調的方法是一樣,就是傳入一個類型,通過Spring SPI的方式查找這個類型的實現類
打個斷點,啟動一下,此時有7個上下文初始器,這是系統(tǒng)自帶的,配置在不同的spring.factories文件中。
現在我要新建一個自己的初始化器
此時為了能夠讓Spring Boot在啟動的時候能夠掃描到我創(chuàng)建的初始化器應該怎么辦?就是在spring.factories文件中添加一下,注冊一下,這樣就能掃描到,這個就是SPI。SPI 全稱為 Service Provider Interface,是一種服務發(fā)現機制。
那么這時候我們再啟動一下Spring Boot,發(fā)現自己創(chuàng)建的ApplicationContextInitializer也已經注冊上來了,變成8個了。
把斷點放掉,在控制臺中也打印出了這句話,那么這個就是第一個擴展點ApplicationContextInitializer
定義了這8個初始化器,那一定是有地方在調它們的,不然怎么會打印出來呢,那具體在什么地方調的,我們在自己的初始化器的地方打斷點,看到已經進來了,然后看下方的堆棧信息,這個就是調用的地方。
原來是調用了run()方法中的prepareContext()方法中的applyInitializers()方法,在這個方法中for循環(huán)的調用各個初始化器的initialize()方法,從而我們就能看到把Jack的ApplicationContextInitializer這句話給打印出來了。
那么這個查找的方法就是以結果為導向來反查調用方,因為你正查的話是很難找到,很難記住的,這個方法希望大家學習到。
那么最后來看下我們第一個擴展點所處的位置
初始化器可以做一些事情,比如Environment對象設置一些變量配置。
2、監(jiān)聽器ApplicationListener
在上面構造函數里我們發(fā)現除了有setInitializers,還有setListeners,那么這個listeners其實也是一個擴展點。
那么什么是監(jiān)聽器,就是這樣的,這個其實就是觀察者模式,ApplicationEventMulticaster發(fā)布事件,各個Listener監(jiān)聽事件。
和初始化器一樣,現在我們自定義兩個監(jiān)聽器,一個是Starting,一個是Started,括號里面的是泛型,這個是一定要寫的,如果不寫的話就是不管什么類型的Event都會監(jiān)聽。
這個泛型是上限為ApplicationEvent類型的Event,具體的實現類有很多個,Starting和Started只是其中兩個。
現在我們還是把這兩個監(jiān)聽器通過SPI的方式加到配置中去
好,運行一下,我們看到這兩句話已經打印出來了
和監(jiān)聽器一樣,既然能夠打印出來,那肯定是有地方在調用,所以我們在JackStartingApplicationListener打個斷點,然后看下堆棧信息
我們可以看到在SpringApplication run()方法的listeners.starting()開始進去發(fā)送ApplicationStartingEvent廣播事件,最后發(fā)布出去,由我們自己編寫的事件監(jiān)聽器接收到。
那么ApplicationStartedEvent事件也是一樣的道理,通過打斷點的方式來找到它的調用方,最后我們再來看下此時的擴展點圖
3、Runner
我們看到在listeners.started()后面有個callRunners
我們點進去看下,它其實就是從容器中獲取兩種類型的Runner,一種是ApplicationRunner,還有一種是CommandLineRunner,然后for循環(huán)的對它們進行調用,那么其實這個也是一個擴展點
我們來寫一個自己的Runner
運行一下,看下打印出來了
那么這個Runner的一般應用場景就是資源釋放清理或者做注冊中心,因為執(zhí)行到Runner的時候項目已經啟動完畢了,這時候就可以注冊到注冊中心上去了。此時我們再看下擴展點圖。
4、BeanFactoryPostProcessor
我們看下run方法里的refreshContext()方法
這個方法底層會調spring里面的refresh()方法,這個方法里面就會做對容器的初始化。紅框里的invokeBeanFactoryPostProcessors()方法,這里也有一個擴展點,就是BeanFactoryPostProcessor,執(zhí)行對BeanFactory的后置處理。Spring Boot解析配置成BeanDefinition的操作也是在此方法中。
現在我們來創(chuàng)建一個自己的BeanFactoryPostProcessor,這個方法里面可以修改beanFactory的屬性,beanfactory里面有BeanDefinition,可以修改BeanDefinition里面的值。BeanDefinition是一個bean的元數據的信息,有多少個bean就有多少個BeanDefinition。
運行一下,也打印出來了
此時我們再看下擴展點圖,越來越完善了。
5、BeanPostProcessor
最后介紹的是BeanPostProcessor,它在通過反射構造函數進行bean實例化之后執(zhí)行,那么紅框里面標出來的registerBeanPostProcessors()方法就是向BeanFactory中注冊beanpostprocessor,用于后續(xù)bean創(chuàng)建的攔截操作。
現在我們來創(chuàng)建一個自己的BeanPostProcessor,一共有兩個方法,postProcessBeforeInitialization和postProcessAfterInitialization,不過我們一般用postProcessAfterInitialization,在bean調用反射構造函數實例化之后執(zhí)行。著名的應用場景AOP底層就是通過BeanPostProcessor來實現的。
現在我在postProcessAfterInitialization上打個斷點,看下堆棧信息是在哪里調用的
是在finishBeanFactoryInitialization()方法處調用的
后記
最后我來把擴展點圖補充完整,如下所示,很清晰明了,在什么時候調用了什么,我們自己開發(fā)的時候結合應用場景,在什么時候要干什么事,就知道要創(chuàng)建什么類型的擴展點了。
本文前三個講的是Spring Boot里面自己有的擴展點,后兩個因為Spring Boot底層調的是Spring的源碼,所以屬于Spring里面的擴展點,所以如果這么算的話Spring里面的擴展點還有很多擴展點,比如InitializeBean、Aware等等這里都沒講,等待大家去發(fā)掘,謝謝觀看 ~
審核編輯:劉清
-
AOP
+關注
關注
0文章
40瀏覽量
11124 -
for循環(huán)
+關注
關注
0文章
61瀏覽量
2549
原文標題:SpingBoot的5個擴展點,超級實用!
文章出處:【微信號:芋道源碼,微信公眾號:芋道源碼】歡迎添加關注!文章轉載請注明出處。
發(fā)布評論請先 登錄
相關推薦
評論