手寫spring筆記

手寫spring筆記

《Spring 手擼專欄》筆記

IoC部分

Bean初始化和屬性注入

Bean的信息封裝在BeanDefinition

/*** 用于記錄Bean的相關信息*/
public class BeanDefinition {/*** Bean對象的類型*/private Class beanClass;/*** Bean對象中的屬性信息*/private PropertyValues propertyValues;/*** 初始化方法的名稱*/private String initMethodName;/*** 銷毀方法的名稱*/private String destroyMethodName;/*** 默認為單例模式*/private String scope;
}

Bean注入過程中有一下幾個主要接口:

  • BeanFactory:Bean工廠接口,其中聲明了獲取Bean對象的方法
  • SingletonBeanRegistry: Bean的單例模式創建接口,聲明了單例對象的創建方法和銷毀方法
  • InstantiationStrategy:實例化Bean對象的策略接口,申明了Bean對象使用何種方式實例化對象的方法,在實現類中使用反射的方式獲取構造方法來得到Bean對象,有JDK和Cglib兩種實現
  • BeanDefinitionRegistry:管理BeanDefinition的注冊,其實現類中包含存有BeanDefinition的Map

其具體流程為:
BeanDefinitionRegistry得到BeanDefinition信息,調用BeanFactorygetBean()方法,判斷是否為單例模式,若為單例模式則調用SingletonBeanRegistrygetSingleton()方法,如果容器中存在Bean則直接返回,不存在則調用InstantiationStrategyinstantiate()方法,使用反射的方式生成Bean對象

在屬性注入中,需要注入的屬性信息封裝在BeanDefinitionPropertyValues中,其本質為一個PropertyValue列表。

在創建Bean對象時,若存在無參構造方法,則使用無參構造方法,若沒有,則從PropertyValues去除構造方法需要的屬性。

/*** Bean對象中的屬性值*/
public class PropertyValue {/*** Bean對象屬性的名稱*/private String name;/*** Bean對象中屬性的實例化對象*/private Object value;
}

資源加載器

Spring需要解析配置文件,對此,定義了一下接口:

  • Resource:資源信息接,用于處理資源加載流,其實現類根據配置文件地址得到資源信息,并向外聲明有得到輸入流的接口,其包含XML文件配置和URL文件配置等方式
  • ResourceLoader:資源加載器接口,通過該接口獲取路徑對應的資源對象,用于獲得資源對象
  • BeanDefinitionReader:BeanDefinition讀取接口,用于讀取配置文件并調用BeanDefinitionRegistry加載BeanDefinition

Bean生命周期

Bean的初始化操作,提供了Bean的初始化和銷毀等方法接口,其包括:

  • InitializingBean:提供了初始化方法的Bean,如果Bean實現了該接口,則會在創建Bean時調用初始化方法
  • DisposableBean:提供了Bean銷毀方法,若實現了該接口,則在容器銷毀時調用其銷毀方法

此外,在BeanDefinition中包含有initMethodNamedestroyMethodName兩個屬性,用于指派Bean的初始化方法和銷毀方法,可以在不實現上述接口的情況下使用反射的方式實現初始化和銷毀方法,在XML中配置init-methoddestroy-method兩個屬性即可

同時,定義有一下兩個接口,來管理Bean的初始化操作

/*** Bean實例化前對其進行預處理的接口,提供修改BeanDefinition的方法*/
public interface BeanFactoryPostProcessor {/*** 所有的BeanDefinition加載完成后而Bean對象實例化之前調用,提供修改BeanDefinition的機制** @param beanFactory* @throws BeansException*/void postProcessorBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException;
}
/*** 對Bean對象初始化前后進行處理的接口,提供Bean初始化前后對其進行操作的方法*/
public interface BeanPostProcessor {/*** 在Bean對象執行初始化前對Bean實例對象進行操作** @param bean     被操作的Bean對象* @param beanName 被操作的Bean對象的名稱* @return* @throws BeansException*/Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException;/*** 在Bean對象執行初始化后對Bean實例對象進行操作** @param bean     被操作的Bean對象* @param beanName 被操作的Bean對象的名稱* @return* @throws BeansException*/Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException;
}

Bean工廠接口

對于復雜類型的Bean,其提供了FactoryBean<T>接口,對于實現了該接口的類,spring在創建Bean對象時,不是直接生成該對象,而是調用getObject方法來生成Bean對象

/*** Bean對象的工廠接口,聲明了從工廠獲得Bean對象的方法*/
public interface BeanFactory {/*** 有參方式獲取Bean** @param name Bean名稱* @return Bean對象實例* @throws BeansException Bean創建時異常*/Object getBean(String name) throws BeansException;/*** 有參方式獲取Bean** @param name Bean名稱* @param args 參數* @return Bean對象實例* @throws BeansException Bean創建時異常*/Object getBean(String name, Object... args) throws BeansException;/*** 按類型獲取Bean對象** @param name         Bean對象名稱* @param requiredType Bean對象類型* @param <T>          Bean對象類型* @return Bean對象實例*/<T> T getBean(String name, Class<T> requiredType) throws BeansException;<T> T getBean(Class<T> requiredType) throws BeansException;
}

Aware感知接口

對于實現了Aware接口的Bean,在spring創建實例時,會將對應的信息注入到Bean中,方便使用者對spring進行自定義擴展開發,其包括如下幾類接口

  • BeanFactoryAware
  • BeanClassLoaderAware
  • BeanNameAware
  • ApplicationContextAware

事件監聽機制

使用觀察者模式,設置了事件監聽機制

  • ApplicationEvent:繼承自EventObject接口,Spring Event事件抽象類,后續的所有事件類都繼承自該類
  • ApplicationListener<E extends ApplicationEvent>:繼承自EventListener接口,其泛型類型為該監聽器關注的事件類型
  • ApplicationEventMulticaster:事件廣播器接口,定義有添加和刪除監聽器以及廣播監聽事件的方法

在使用過程中,首先調用ApplicationEventMulticasteraddApplicationListener方法將自定義監聽器加入到廣播器中,當要發布事件時,調用multicastEvent方法,該方法會便利已注冊的所有監聽器,并向關注該事件的監聽器發送通知,調用其onApplicationEvent方法

應用上下文

接口ApplicationContext整合了上述的各種方法,并提供了向外的操作接口。其中,ApplicationContext的實現類中定義了核心方法refresh(),其具體內容如下:

    @Overridepublic void refresh() throws BeansException {// 創建BeanFactory,并加載BeanDefinitionrefreshBeanFactory();// 獲取BeanFactoryConfigurableListableBeanFactory beanFactory = getBeanFactory();// 添加ApplicationContextAwareProcessor對象處理ApplicationContextAwarebeanFactory.addBeanPostProcessor(new ApplicationContextAwareProcessor(this));// 在Bean實例化之前,執行BeanFactoryPostProcessor方法invokeBeanFactoryPostProcessors(beanFactory);// 將BeanPostProcessor提前注冊到容器中registerBeanPostProcessors(beanFactory);// 提前實例化單例Bean對象beanFactory.preInstantiateSingletons();// 初始化事件發布者initApplicationEventMulticaster();// 注冊事件監聽器registerListeners();// 發布容器刷新完成事件finishRefresh();}

三級緩存機制

為了解決循環依賴,spring使用了三級緩存,如下所示:

    /*** Bean容器一級緩存,用于存儲Bean的完整實例化對象*/private Map<String, Object> singletonObjects = new HashMap<>();/*** Bean容器二級緩存,用于存儲Bean的早期非完整實例化對象*/private Map<String, Object> earlySingletonObjects = new HashMap<>();/*** Bean容器三級緩存,用于存儲Bean的代理對象*/private Map<String, ObjectFactory<?>> singletonFactories = new HashMap<>();/**

其中,三級緩存存儲的不是Bean對象,而是一個工廠對象,其目的是為了提早暴露Bean,等到后續的Bean依賴該Bean的時候,就會調用工廠的getObject方法獲取到Bean對象。三級緩存的順序如下:

    @Overridepublic Object getSingleton(String beanName) throws BeansException {Object singletonObject = singletonObjects.get(beanName);if (singletonObject == null) {// 如果沒有完整對象,則從二級緩存中獲取未完整實例化的對象singletonObject = earlySingletonObjects.get(beanName);if (singletonObject == null) {// 如果二級緩存中也沒有,則從代理對象處生成未完成的實例化對象,并刪除代理對象ObjectFactory<?> singletonFactory = singletonFactories.get(beanName);if (singletonFactory != null) {singletonObject = singletonFactory.getObject();earlySingletonObjects.put(beanName, singletonObject);singletonFactories.remove(beanName);}}}return singletonObject;}

注解配置信息

這里給出了一下幾個注解功能的實現:

  • @Component:將類注入到容器中
  • @Autowired:對Bean對象對應的該屬性進行依賴注入
  • @Scope:配置該類的作用域為單例還是原型
  • @Value:注入配置文件中的信息

通過讀取配置文件中<context:component-scan />指定的路徑,對其下所有的類進行掃描,若標注有@Component注解,則生成對應的BeanDefinition,并讀取其屬性,生成對應的PropertyValue,若屬性標注有@Value注解,表示屬性值為配置文件中的對應數值,若標注有@Autowired注解,則表示屬性值為Bean。生成好BeanDefinition后,就和原本流程一樣了

AOP

切點和匹配接口定義

AOP的核心是使用代理的方式生成代理對象,其核心接口包括:

  • ClassFilter:定義類匹配類,用于切點找到給定的接口和目標類,提供了判斷切入點是否應用在給定的接口或目標類中的方法
  • MethodMatcher:方法匹配類,找到表達式范圍內匹配下的目標類和方法,提供了判斷給定的方法是否匹配表達式的方法
  • Pointcut:切入點接口,定義用于獲取ClassFilterMethodMatcher的兩個類
  • AopProxy:代理接口,用于獲取代理類

使用了JDK和AspectJ兩種方式實現了上述接口,并通過ProxyFactory工廠類來對兩種實現的選擇進行了封裝

AOP融入Bean生命周期

將AOP融入Bean的聲明周期,其利用了BeanPostProcessor接口,使用該接口的postProcessAfterInitialization對實例化后的原對象進行包裝,返回其代理對象

其定義了Advisor訪問者接口,整合切面pointcut和攔截器advice

對于反射要用的advice接口,定義了繼承自該接口的BeforeAdvcice等接口,提供前置方法等AOP方法接口。同樣地,對于攔截器MethodInterceptor,也封裝了對應的MethodBeforeAdviceInterceptor等類,來實現各種AOP操作

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/41021.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/41021.shtml
英文地址,請注明出處:http://en.pswp.cn/news/41021.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

MFC第三十天 通過CToolBar類開發文字工具欄和工具箱、GDI+邊框填充以及基本圖形的繪制方法、圖形繪制過程的反色線模型和實色模型

文章目錄 CControlBar通過CToolBar類開發文字工具欄和工具箱CMainFrame.hCAppCMainFrm.cppCMainView.hCMainView.cppCEllipse.hCEllipse.cppCLine.hCLine.cppCRRect .hCRRect .cpp CControlBar class AFX_NOVTABLE CControlBar : public CWnd{DECLARE_DYNAMIC(CControlBar)pro…

OC調用Swift編寫的framework

一、前言 隨著swift趨向穩定&#xff0c;越來越多的公司都開始用swift來編寫蘋果相關的業務了&#xff0c;關于swift的利弊這里就不多說了。這里詳細介紹OC調用swift編寫的framework庫的步驟 二、制作framework 1、新建項目&#xff0c;選擇framework 2、填寫framework的名稱…

AutoHotkey:定時刪除目錄下指定分鐘以前的文件,帶UI界面

刪除指定目錄下&#xff0c;所有在某個指定分鐘以前的文件&#xff0c;可以用來清理經常生成很多文件的目錄&#xff0c;但又需要保留最新的一部分文件 支持拖放目錄到界面 能夠記憶設置&#xff0c;下次啟動后不用重新設置&#xff0c;可以直接開始 應用場景比如&#xff1a…

WinForm內嵌Unity3D

Unity3D可以C#腳本進行開&#xff0c;使用vstu2013.msi插件&#xff0c;可以實現在VS2013中的調試。在開發完成后&#xff0c;由于項目需要&#xff0c;需要將Unity3D嵌入到WinForm中。WinForm中的UnityWebPlayer Control可以載入Unity3D。先看效果圖。 一、為了能夠動態設置ax…

【boost網絡庫從青銅到王者】第五篇:asio網絡編程中的同步讀寫的客戶端和服務器示例

文章目錄 1、簡介2、客戶端設計3、服務器設計3.1、session函數3.2、StartListen函數3、總體設計 4、效果測試5、遇到的問題5.1、服務器遇到的問題5.1.1、不用顯示調用bind綁定和listen監聽函數5.1.2、出現 Error occured!Error code : 10009 .Message: 提供的文件句柄無效。 [s…

Failed to execute goal org.apache.maven.plugins

原因&#xff1a; 這個文件D:\java\maven\com\ruoyi\pg-student\maven-metadata-local.xml出了問題 解決&#xff1a; 最簡單的直接刪除D:\java\maven\com\ruoyi\pg-student\maven-metadata-local.xml重新打包 或者把D:\java\maven\com\ruoyi\pg-student這個目錄下所有文件…

性能測試場景設計

性能測試場景設計&#xff0c;是性能測試中的重要概念&#xff0c;性能測試場景設計&#xff0c;目的是要描述如何執行性能測試。 通常來講&#xff0c;性能測試場景設計主要會涉及以下部分&#xff1a; 并發用戶數是多少&#xff1f; 測試剛開始時&#xff0c;以什么樣的速率…

Spring WebFlux 的詳細介紹

Spring WebFlux 是基于響應式編程的框架&#xff0c;用于構建異步、非阻塞的 Web 應用程序。它是Spring框架的一部分&#xff0c;專注于支持響應式編程范式&#xff0c;使應用程序能夠高效地處理大量的并發請求和事件。 以下是關于 Spring WebFlux 的詳細介紹&#xff1a; 1. *…

go入門實踐五-實現一個https服務

文章目錄 前言生成證書申請免費的證書使用Go語言生成自簽CA證書 https的客戶端和服務端服務端代碼客戶端代碼 tls的客戶端和服務端服務端客戶端 前言 在公網中&#xff0c;我想加密傳輸的數據。(1)很自然&#xff0c;我想到了把數據放到http的請求中&#xff0c;然后通過tls確…

2023年京東寵物食品行業數據分析(京東大數據)

寵物食品市場需求主要來自于養寵規模&#xff0c;近年來由于我國寵物數量及養寵人群的規模均在不斷擴大&#xff0c;寵物相關產業和市場規模也在蓬勃發展&#xff0c;寵物食品市場也同樣保持正向增長。 根據鯨參謀電商數據分析平臺的相關數據顯示&#xff0c;2023年1月-7月&am…

vue5種模糊查詢方式

在Vue中&#xff0c;有多種方式可以實現模糊查詢。以下是五種常見的模糊查詢方式&#xff1a; 使用JavaScript的filter()方法&#xff1a;使用filter()方法可以對數組進行篩選&#xff0c;根據指定的條件進行模糊查詢。例如&#xff1a; data() {return {items: [{ name: App…

接口自動化測試(添加課程接口調試,調試合同上傳接口,合同列表查詢接口,批量執行)

1、我們把信息截取一下 1.1 添加一個新的請求 1.2 對整個請求進行保存&#xff0c;Ctrl S 2、這一次我們添加的是課程添加接口&#xff0c;以后一個接口完成&#xff0c;之后Ctrl S 就能夠保存 2.1 選擇方法 2.2 設置請求頭&#xff0c;參數數據后期我們通過配置設置就行 3、…

收銀一體化-億發2023智慧門店新零售營銷策略,實現全渠道運營

伴隨著互聯網電商行業的興起&#xff0c;以及用戶理念的改變&#xff0c;大量用戶從線下涌入線上&#xff0c;傳統的線下門店人流量急劇收縮&#xff0c;門店升級幾乎成為了每一個零售企業的發展之路。智慧門店新零售收銀解決方案是針對傳統零售企業面臨的諸多挑戰和問題&#…

Mathematica 與 Matlab 常見復雜指令集匯編

Mathematica 常見指令匯編 Mathematica 常見指令 NDSolve 求解結果的保存 sol NDSolve[{y[x] x^2, y[0] 0, g[x] -y[x]^2, g[0] 1}, {y, g}, {x, 0, 1}]; numericSoly sol[[1, 1, 2]]; numericSolg sol[[1, 2, 2]]; data Table[{x, numericSoly[x], numericSolg[x]},…

JVM——類加載器

回顧一下類加載過程 類加載過程&#xff1a;加載->連接->初始化。連接過程又可分為三步:驗證->準備->解析。 一個非數組類的加載階段&#xff08;加載階段獲取類的二進制字節流的動作&#xff09;是可控性最強的階段&#xff0c;這一步我們可以去完成還可以自定義…

【計算機網絡篇】UDP協議

?作者簡介&#xff1a;大家好&#xff0c;我是小楊 &#x1f4c3;個人主頁&#xff1a;「小楊」的csdn博客 &#x1f433;希望大家多多支持&#x1f970;一起進步呀&#xff01; UDP協議 1&#xff0c;UDP 簡介 UDP&#xff08;User Datagram Protocol&#xff09;是一種無連…

Flink學習筆記(一)

流處理 批處理應用于有界數據流的處理&#xff0c;流處理則應用于無界數據流的處理。 有界數據流&#xff1a;輸入數據有明確的開始和結束。 無界數據流&#xff1a;輸入數據沒有明確的開始和結束&#xff0c;或者說數據是無限的&#xff0c;數據通常會隨著時間變化而更新。 在…

Kaptcha的基本應用

Kaptcha Kaptcha 是一個用于生成和驗證驗證碼的 Java 庫&#xff0c;提供了豐富的生成和驗證功能&#xff0c;并支持自定義配置。它可以用于增加應用程序的安全性&#xff0c;防止機器人和惡意攻擊。 Kaptcha 可以生成各種類型的驗證碼&#xff0c;包括數字、字母、數字字母組…

KDD 2023 獲獎論文公布,港中文、港科大等獲最佳論文獎

ACM SIGKDD&#xff08;國際數據挖掘與知識發現大會&#xff0c;KDD&#xff09;是數據挖掘領域歷史最悠久、規模最大的國際頂級學術會議&#xff0c;也是首個引入大數據、數據科學、預測分析、眾包等概念的會議。 今年&#xff0c;第29屆 KDD 大會于上周在美國加州長灘圓滿結…

HTTP--Request詳解

請求消息數據格式 請求行 請求方式 請求url 請求協議/版本 GET /login.html HTTP/1.1 請求頭 客戶端瀏覽器告訴服務器一些信息 請求頭名稱: 請求頭值 常見的請求頭&#xff1a; User-Agent&#xff1a;瀏覽器告訴服務器&#xff0c;我訪問你使用的瀏覽器版本信息 可…