前言
Spring是什麼?
Spring發展至今,已經不是簡單的Spring Framework,它包括Spring Data、Spring Boot、Spring Cloud等等組成
不過這裡我們僅討論Spring Framework,重點討論Spring的啟動過程及擴充應用
Spring啟動
監聽器啟動
web.xml配置
data:image/s3,"s3://crabby-images/b1590/b159070b7439019efbe7f038cfba7cc0d2bbc174" alt="Spring啟動過程——原始碼分析"
ContextLoaderListener實現了ServletContextListener,會在web容器啟動的時候呼叫
data:image/s3,"s3://crabby-images/ec7b5/ec7b5e6d9b521e041d0872551771a17326a5be40" alt="Spring啟動過程——原始碼分析"
data:image/s3,"s3://crabby-images/52028/520289d2c9e705549603b64251ddd98530d997dd" alt="Spring啟動過程——原始碼分析"
data:image/s3,"s3://crabby-images/421f7/421f77ec6f7a79c29f2bcbe81c7ba745370b52c0" alt="Spring啟動過程——原始碼分析"
而後進入configureAndRefreshWebApplicationContext(cwac, servletContext)
data:image/s3,"s3://crabby-images/caad3/caad323152a933c78c3862e2bfdfb030de0b2523" alt="Spring啟動過程——原始碼分析"
Spring容器建立過程
refresh()
刪減了一些無關緊要的程式碼
@Override
public void refresh() throws BeansException, IllegalStateException {
synchronized (this.startupShutdownMonitor) {
// Prepare this context for refreshing.
prepareRefresh();
// Tell the subclass to refresh the internal bean factory.
ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();
// Prepare the bean factory for use in this context.
prepareBeanFactory(beanFactory);
try {
// Allows post-processing of the bean factory in context subclasses.
postProcessBeanFactory(beanFactory);
// Invoke factory processors registered as beans in the context.
invokeBeanFactoryPostProcessors(beanFactory)
// Register bean processors that intercept bean creation.
registerBeanPostProcessors(beanFactory);
// Initialize message source for this context.
initMessageSource();
// Initialize event multicaster for this context.
initApplicationEventMulticaster();
// Initialize other special beans in specific context subclasses.
onRefresh();
// Check for listener beans and register them.
registerListeners();
// Instantiate all remaining (non-lazy-init) singletons.
finishBeanFactoryInitialization(beanFactory);
// Last step: publish corresponding event.
finishRefresh();
}
}
}
複製程式碼
prepareRefresh()
容器refresh預準備工作 AbstractApplicationContext
data:image/s3,"s3://crabby-images/6babf/6babf0d0da701259a2caf009446a439af1ee16ee" alt="Spring啟動過程——原始碼分析"
-
initPropertySources() AbstractApplicationContext子類自定義屬性設定的方法
-
getEnvironment().validateRequiredProperties() 校驗設定屬性
-
this.earlyApplicationEvents = new LinkedHashSet<>() 儲存容器早期的事件
obtainFreshBeanFactory()
獲取工廠
data:image/s3,"s3://crabby-images/d2542/d2542ec92e468bebc529173499507f7625a59ba1" alt="Spring啟動過程——原始碼分析"
- refreshBeanFactory()
data:image/s3,"s3://crabby-images/c60f3/c60f33b371e49fded30e605ead8772518b89036f" alt="Spring啟動過程——原始碼分析"
BeanDefinition是工廠中非常重要的屬性,它將所有需要spring管理的bean資訊都儲存起來,會在後邊一個一個去例項化
最終進入
data:image/s3,"s3://crabby-images/cc1fb/cc1fb6b5dcaa8edcd9dff71cb4dedfc610b52794" alt="Spring啟動過程——原始碼分析"
根據件用xmlReader去讀取一個個配置檔案 最終呼叫
this.beanDefinitionMap.put(beanName, beanDefinition)
refreshBeanFactory執行完進入getBeanFactory()
直接返回改beanFactory 方法結束
prepareBeanFactory(beanFactory)
data:image/s3,"s3://crabby-images/e7fa5/e7fa532240dddb134b278530ecbb41fe1e67a8ee" alt="Spring啟動過程——原始碼分析"
-
在工廠設定類載入器、表示式解析器
-
給工廠新增ApplicationContextAwareProcessor(processor先不解釋有什麼意思 後續會說明)
-
設定需要忽略的自動裝配的介面(EnvironmentAware、ApplicationContextAware等Aware)
-
給工廠新增 beanFactory.addBeanPostProcessor(new ApplicationListenerDetector(this))
接著上面方法
data:image/s3,"s3://crabby-images/f151a/f151a783736ef0e5e1e6483abfab1d8415aa0944" alt="Spring啟動過程——原始碼分析"
data:image/s3,"s3://crabby-images/d4977/d49776ee5bbb01b1a530792982bb935b4e78939d" alt="Spring啟動過程——原始碼分析"
- environment(ConfigurableEnvironment)
- systemProperties(Map<String, Object>)
- systemEnvironment(Map<String, Object>)
postProcessBeanFactory(beanFactory)
BeanFactory準備工作完成後 後置處理
空方法 子類自定義屬性設定的方法
這裡web環境 用的工廠是AbstractRefreshableWebApplicationContext
記住這裡只是web端擴充才有的實現
data:image/s3,"s3://crabby-images/3779e/3779e476bb8cbdc2592bdff1881f37e2ebb188a3" alt="Spring啟動過程——原始碼分析"
WebApplicationContextUtils.registerWebApplicationScopes(beanFactory, this.servletContext)
新增了幾個scope
spring本身只有singleton和prototype兩個,web環境加上了request、session、globalSession
invokeBeanFactoryPostProcessors(beanFactory)
這裡沒仔細看
data:image/s3,"s3://crabby-images/a22b4/a22b467c34128dc7b7cd3baac6c4966185492e7a" alt="Spring啟動過程——原始碼分析"
BeanFactoryPostProcessor、BeanDefinitionRegistryPostProcessor
他們的關係:BeanDefinitionRegistryPostProcessor extends BeanFactoryPostProcessor
data:image/s3,"s3://crabby-images/c09d4/c09d47c738d1ad1d48c1fe097a697c8626a6a0bb" alt="Spring啟動過程——原始碼分析"
data:image/s3,"s3://crabby-images/74ea6/74ea6c0f287f77e9bb13c45b669597fe8e1a6861" alt="Spring啟動過程——原始碼分析"
data:image/s3,"s3://crabby-images/a3f9e/a3f9e01a4bcb2b1f0478aa80fbd20cccac9514b3" alt="Spring啟動過程——原始碼分析"
data:image/s3,"s3://crabby-images/e72a0/e72a06c9fd9a6fa884c4c6e4d50b14b0ca8609e3" alt="Spring啟動過程——原始碼分析"
data:image/s3,"s3://crabby-images/222f9/222f9fb95a8a7f9845fc8fc192d2a203e46af3ce" alt="Spring啟動過程——原始碼分析"
-
先獲取所有的BeanDefinitionRegistryPostProcessor
-
依次執行實現了PriorityOrdered、Ordered介面和沒有實現它們的 processor postProcessor.postProcessBeanDefinitionRegistry(registry)
在這裡我debug發現 spring內部有一個ConfigurationClassPostProcessor
它的作用是負責解析處理所有@Configuration標籤類,並將Bean定義(包括其它processor)註冊到BeanFactory中。
這裡註冊完之後 再次獲取所有BeanDefinitionRegistryPostProcessor,按順序執行invokeBeanDefinitionRegistryPostProcessors(currentRegistryProcessors, registry)
重點:其它的BeanDefinitionRegistryPostProcessor就是Spring提供給我們註冊bean的通道 -
再執行postProcessor.postProcessBeanFactory(beanFactory)
-
獲取所有的BeanFactoryPostProcessor
-
分類執行 執行過的不再執行
data:image/s3,"s3://crabby-images/de314/de3149de0f47a12ecd960a87596b3bfa2ee41b4c" alt="Spring啟動過程——原始碼分析"
實踐出真知 這裡是用SpringBoot簡單搭起的一個demo,跟上面不一樣,不過不影響測試結果
data:image/s3,"s3://crabby-images/faa96/faa96e3a0c88cce43092c670b26adf9d1d6fa1e0" alt="Spring啟動過程——原始碼分析"
data:image/s3,"s3://crabby-images/61c5a/61c5ab2f9a9602840f5282ca4389c040dbb9cfea" alt="Spring啟動過程——原始碼分析"
data:image/s3,"s3://crabby-images/14e9f/14e9f648a6d5ed871c8a6b200f16d1e6cdcade35" alt="Spring啟動過程——原始碼分析"
data:image/s3,"s3://crabby-images/9145f/9145f89b8ea2cf4721798caed32169b44c253b4c" alt="Spring啟動過程——原始碼分析"
- 先執行BeanDefinitionRegistryPostProcessor的postProcessBeanDefinitionRegistry
- 再執行BeanDefinitionRegistryPostProcessor的postProcessBeanFactory
- 最後執行BeanFactoryPostProcessor的postProcessBeanFactory
registerBeanPostProcessors(beanFactory)
註冊Bean的後置處理器(跟上面的FactoryPostProcessor不一樣)
Bean的生命週期: Bean建立--初始化--銷燬
不同BeanPostProcessor功能都不一樣
BeanPostProcessor的實現子介面:
DestructionAwareBeanPostProcessor InstantiationAwareBeanPostProcessor MergedBeanDefinitionPostProcessor SmartInstantiationAwareBeanPostProcessor
BeanPostProcessor
data:image/s3,"s3://crabby-images/3143f/3143f2cb8a21a5b5e9f0519da856ba5776d95b8b" alt="Spring啟動過程——原始碼分析"
data:image/s3,"s3://crabby-images/80140/80140d9049ddc6d40ab644b601132ea2593d1aa6" alt="Spring啟動過程——原始碼分析"
獲取所有BeanPostProcessor 依次註冊了PriorityOrdered、Ordered介面、沒有實現它們的、MergedBeanDefinitionPostProcessor
註冊呼叫的是
data:image/s3,"s3://crabby-images/ece2d/ece2d520756123731f363d1b3fa875f0e57d55cf" alt="Spring啟動過程——原始碼分析"
註冊ApplicationListenerDetector :在bean初始化後 檢查如果是listener的話 執行applicationContext.addApplicationListener
initMessageSource()
初始化MessageSource元件(做國際化、訊息繫結、訊息解析)
data:image/s3,"s3://crabby-images/9d1a9/9d1a9fd5a0d38b34a4de3204aed70d8390f5493f" alt="Spring啟動過程——原始碼分析"
- 獲取BeanFactory
- 檢視容器是否有messageSource的元件
-
有->this.applicationEventMulticaster beanFactory.getBean(APPLICATION_EVENT_MULTICASTER_BEAN_NAME, ApplicationEventMulticaster.class)
-
沒有->new DelegatingMessageSource() 註冊一個MessageSource元件
beanFactory.registerSingleton(MESSAGE_SOURCE_BEAN_NAME, this.messageSource)
initApplicationEventMulticaster()
初始化事件派發器
data:image/s3,"s3://crabby-images/7ba28/7ba2887861c13573c3187a69d0e266437109a807" alt="Spring啟動過程——原始碼分析"
- 獲取BeanFactory
- 檢視容器是否有applicationEventMulticaster的元件
-
有->this.messageSource = beanFactory.getBean(MESSAGE_SOURCE_BEAN_NAME, MessageSource.class)
-
沒有->new SimpleApplicationEventMulticaster() beanFactory.registerSingleton(APPLICATION_EVENT_MULTICASTER_BEAN_NAME, this.applicationEventMulticaster)
這個派發器在後續很有用
onRefresh()
子類自定義的方法 web環境下直接進入一個空方法
還不確定什麼環境下會有執行 歡迎聯絡交流~
registerListeners()
data:image/s3,"s3://crabby-images/585d5/585d5c0c6905dfaae3309e07d7255dab04aff823" alt="Spring啟動過程——原始碼分析"
- 獲取所有ApplicationListener
- getApplicationEventMulticaster().addApplicationListenerBean
這裡拿到的派發器就是剛才註冊的
finishBeanFactoryInitialization(beanFactory)
較為複雜,單獨作為一篇
finishRefresh()
data:image/s3,"s3://crabby-images/db421/db4215ea917ec0a980d409c754b10fbed101c4fe" alt="Spring啟動過程——原始碼分析"
- 初始化和生命週期有關的後置處理器 initLifecycleProcessor()
- LifecycleProcessor:
- void onRefresh();
呼叫生命週期處理器的onRefresh方法,這個方法會找出Spring容器中實現了SmartLifecycle介面的類並進行start方法的呼叫
- 釋出ContextRefreshedEvent事件告知對應的ApplicationListener進行響應的操作
publishEvent(new ContextRefreshedEvent(this)) - 呼叫LiveBeansView的registerApplicationContext方法:如果設定了JMX相關的屬性,則就呼叫該方法 LiveBeansView.registerApplicationContext(this)