Spring致力於提供一種方法管理你的業務對象。在大量Java EE的應用中,隨處可見Spring。今天我將簡單的介紹一下Spring這個框架。php
本文適合讀者:java
想學Spring的Java開發者web
剛用Spring不久的人正則表達式
爲何要使用Spring?spring
Spring主要兩個有功能爲咱們的業務對象管理提供了很是便捷的方法:編程
DI(Dependency Injection,依賴注入)安全
AOP(Aspect Oriented Programming,面向切面編程)多線程
每個類實現了Bean的規範才能夠由Spring來接管,那麼Bean的規範是什麼呢?架構
必須是個公有(public)類併發
有無參構造函數
用公共方法暴露內部成員屬性(getter,setter)
實現這樣規範的類,被稱爲Java Bean。便是一種可重用的組件。
簡單來講,一個系統中可能會有成千上萬個對象。若是要手工維護它們之間的關係,這是不可想象的。咱們能夠在Spring的XML文件描述它們之間的關係,由Spring自動來注入它們——好比A類的實例須要B類的實例做爲參數set進去。
就以日誌系統爲例。在執行某個操做先後都須要輸出日誌,若是手工加代碼,那簡直太可怕了。並且等代碼龐大起來,也是很是難維護的一種狀況。這裏就須要面向切面來編程
如你所見,在bean準備就緒以前,bean工廠執行了若干啓動步驟。咱們對圖進行詳細描述:
Spring對bean進行實例化;
Spring將值和bean的引用注入到bean對應的屬性中;
若是bean實現了BeanNameAware接口,Spring將bean的ID傳遞給setBean-Name()方法;
若是bean實現了BeanFactoryAware接口,Spring將調用setBeanFactory()方法,將BeanFactory容器實例傳入;
若是bean實現了ApplicationContextAware接口,Spring將調用setApplicationContext()方法,將bean所在的應用上下文的引用傳入進來;
若是bean實現了BeanPostProcessor接口,Spring將調用它們的post-ProcessBeforeInitialization()方法;
若是bean實現了InitializingBean接口,Spring將調用它們的after-PropertiesSet()方法。相似地,若是bean使用init-method聲明瞭初始化方法,該方法也會被調用;
若是bean實現了BeanPostProcessor接口,Spring將調用它們的post-ProcessAfterInitialization()方法;
此時,bean已經準備就緒,能夠被應用程序使用了,它們將一直駐留在應用上下文中,直到該應用上下文被銷燬;
若是bean實現了DisposableBean接口,Spring將調用它的destroy()接口方法。一樣,若是bean使用destroy-method聲明瞭銷燬方法,該方法也會被調用。
Spring定義了多種Bean做用域,能夠基於這些做用域建立bean,包括:
單例(Singleton):在整個應用中,只建立bean的一個實例。
原型(Prototype):每次注入或者經過Spring應用上下文獲取的時候,都會建立一個新的bean實例。
會話(Session):在Web應用中,爲每一個會話建立一個bean實例。
請求(Rquest):在Web應用中,爲每一個請求建立一個bean實例。
在代碼裏看起來是這樣的:
@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)public class MyIsBean{...}
XML版本:
<bean id="BEANID" class = "com.my.beans" scope="prototype">
在默認狀況下,Spring應用上下文中全部bean都是做爲以單例(singleton)的形式建立的。也就是說,無論給定的一個bean被注入到其餘bean多少次,每次所注入的都是同一個實例。
在大多數狀況下,單例bean是很理想的方案。初始化和垃圾回收對象實例所帶來的成本只留給一些小規模任務,在這些任務中,讓對象保持無狀態而且在應用中反覆重用這些對象可能並不合理。
有時候,可能會發現,你所使用的類是易變的(mutable),它們會保持一些狀態,所以重用是不安全的。在這種狀況下,將class聲明爲單例的bean就不是什麼好主意了,由於對象會被污染,稍後重用的時候會出現意想不到的問題。
如下是聲明Bean的註解:
@Component 組件,沒有明確的角色
@Service 在業務邏輯層使用
@Repository 在數據訪問層使用
@Controller 在展示層使用(MVC -> Spring MVC)使用
在這裏,能夠指定bean的id名:Component(「yourBeanName」)
同時,Spring支持將@Named做爲@Component註解的替代方案。二者之間有一些細微的差別,可是在大多數場景中,它們是能夠互相替換的。
不只僅是對象,還有在構造器上,還能用在屬性的Setter方法上。
無論是構造器、Setter方法仍是其餘的方法,Spring都會嘗試知足方法參數上所聲明的依賴。假若有且只有一個bean匹配依賴需求的話,那麼這個bean將會被裝配進來。
若是沒有匹配的bean,那麼在應用上下文建立的時候,Spring會拋出一個異常。爲了不異常的出現,你能夠將@Autowired的required屬性設置爲false。
將required屬性設置爲false時,Spring會嘗試執行自動裝配,可是若是沒有匹配的bean的話,Spring將會讓這個bean處於未裝配的狀態。可是,把required屬性設置爲false時,你須要謹慎對待。若是在你的代碼中沒有進行null檢查的話,這個處於未裝配狀態的屬性有可能會出現NullPointerException。
@Inject註解來源於Java依賴注入規範,該規範同時還爲咱們定義了@Named註解。在自動裝配中,Spring同時支持@Inject和@Autowired。儘管@Inject和@Autowired之間有着一些細微的差異,可是在大多數場景下,它們都是能夠互相替換的。
@Autowired
是最多見的註解之一,但在老項目中,你可能會看到這些註解,它們的做用和@Autowired
相近:
@Inject
是JSR-330提供的註解
@Resource
是JSR-250提供的註解
假設你但願一個或多個bean只有在應用的類路徑下包含特定的庫時才建立。或者咱們但願某個bean只有當另外某個特定的bean也聲明瞭以後纔會建立。咱們還可能要求只有某個特定的環境變量設置以後,纔會建立某個bean。
在Spring 4以前,很難實現這種級別的條件化配置,可是Spring 4引入了一個新的@Conditional
註解,它能夠用到帶有@Bean註解的方法上。若是給定的條件計算結果爲true,就會建立這個bean,不然的話,這個bean會被忽略。
經過ConditionContext,咱們能夠作到以下幾點:
藉助getRegistry()返回的BeanDefinitionRegistry檢查bean定義;
藉助getBeanFactory()返回的ConfigurableListableBeanFactory檢查bean是否存在,甚至探查bean的屬性;
藉助getEnvironment()返回的Environment檢查環境變量是否存在以及它的值是什麼;
讀取並探查getResourceLoader()返回的ResourceLoader所加載的資源;
藉助getClassLoader()返回的ClassLoader加載並檢查類是否存在。
在聲明bean的時候,經過將其中一個可選的bean設置爲首選(primary)bean可以避免自動裝配時的歧義性。當遇到歧義性的時候,Spring將會使用首選的bean,而不是其餘可選的bean。實際上,你所聲明就是「最喜歡」的bean。
設置首選bean的侷限性在於@Primary沒法將可選方案的範圍限定到惟一一個無歧義性的選項中。它只能標示一個優先的可選方案。當首選bean的數量超過一個時,咱們並無其餘的方法進一步縮小可選範圍。
與之相反,Spring的限定符可以在全部可選的bean上進行縮小範圍的操做,最終可以達到只有一個bean知足所規定的限制條件。若是將全部的限定符都用上後依然存在歧義性,那麼你能夠繼續使用更多的限定符來縮小選擇範圍。
@Qualifier註解是使用限定符的主要方式。它能夠與@Autowired和@Inject協同使用,在注入的時候指定想要注入進去的是哪一個bean。例如,咱們想要確保要將IceCream注入到setDessert()之中:
@Autowired@Qualifier("iceCream")public void setDessert(Dessert dessert){ this.dessert = dessert; }
這是使用限定符的最簡單的例子。爲@Qualifier註解所設置的參數就是想要注入的bean的ID。全部使用@Component註解聲明的類都會建立爲bean,而且bean的ID爲首字母變爲小寫的類名。所以,@Qualifier(「iceCream」)指向的是組件掃描時所建立的bean,而且這個bean是IceCream類的實例。
實際上,還有一點須要補充一下。更準確地講,@Qualifier(「iceCream」)所引用的bean要具備String類型的「iceCream」做爲限定符。若是沒有指定其餘的限定符的話,全部的bean都會給定一個默認的限定符,這個限定符與bean的ID相同。所以,框架會將具備「iceCream」限定符的bean注入到setDessert()方法中。這恰巧就是ID爲iceCream的bean,它是IceCream類在組件掃描的時候建立的。
基於默認的bean ID做爲限定符是很是簡單的,但這有可能會引入一些問題。若是你重構了IceCream類,將其重命名爲Gelato的話,那此時會發生什麼狀況呢?若是這樣的話,bean的ID和默認的限定符會變爲gelato,這就沒法匹配setDessert()方法中的限定符。自動裝配會失敗。
這裏的問題在於setDessert()方法上所指定的限定符與要注入的bean的名稱是緊耦合的。對類名稱的任意改動都會致使限定符失效。
Value實現資源的注入
Java配置方式:initMethod和destoryMethod
註解:@PostConstruct和@PreDestory
提供在不一樣的環境下使用不一樣的配置
激活Profile
Spring在肯定哪一個profile處於激活狀態時,須要依賴兩個獨立的屬性:spring.profiles.active和spring.profiles.default。若是設置了spring.profiles.active屬性的話,那麼它的值就會用來肯定哪一個profile是激活的。但若是沒有設置spring.profiles.active屬性的話,那Spring將會查找spring.profiles.default的值。若是spring.profiles.active和spring.profiles.default均沒有設置的話,那就沒有激活的profile,所以只會建立那些沒有定義在profile中的bean。
使用profile進行測試
當運行集成測試時,一般會但願採用與生產環境(或者是生產環境的部分子集)相同的配置進行測試。可是,若是配置中的bean定義在了profile中,那麼在運行測試時,咱們就須要有一種方式來啓用合適的profile。
Spring提供了@ActiveProfiles註解,咱們可使用它來指定運行測試時要激活哪一個profile。在集成測試時,一般想要激活的是開發環境的profile。
好比Profile(「dev」)
使用Application Event
能夠作到Bean與Bean之間的通訊
Spring的事件須要遵循以下流程:
自定義事件,集成ApplicationEvent
定義事件監聽器,實現ApplicationListener
使用容器發佈事件
通知定義了切面是什麼以及什麼時候使用。除了描述切面要完成的工做,通知還解決了什麼時候執行這個工做的問題。它應該應用在某個方法被調用以前?以後?以前和以後都調用?仍是隻在方法拋出異常時調用?
Spring切面能夠應用5種類型的通知:
前置通知(Before):在目標方法被調用以前調用通知功能;
後置通知(After):在目標方法完成以後調用通知,此時不會關心方法的輸出是什麼;
返回通知(After-returning):在目標方法成功執行以後調用通知;
異常通知(After-throwing):在目標方法拋出異常後調用通知;
環繞通知(Around):通知包裹了被通知的方法,在被通知的方法調用以前和調用以後執行自定義的行爲。
對應註解:
注 解 | 通 知 |
---|---|
@After | 通知方法會在目標方法返回或拋出異常後調用 |
@AfterReturning | 通知方法會在目標方法返回後調用 |
@AfterThrowing | 通知方法會在目標方法拋出異常後調用 |
@Around | 通知方法會將目標方法封裝起來 |
@Before | 通知方法會在目標方法調用以前執行 |
鏈接點是在應用執行過程當中可以插入切面的一個點。這個點能夠是調用方法時、拋出異常時、甚至修改一個字段時。切面代碼能夠利用這些點插入到應用的正常流程之中,並添加新的行爲。
若是說通知定義了切面的「什麼」和「什麼時候」的話,那麼切點就定義了「何處」 。切點的定義會匹配通知所要織入的一個或多個鏈接點。咱們一般使用明確的類和方法名稱,或是利用正則表達式定義所匹配的類和方法名稱來指定這些切點。有些AOP框架容許咱們建立動態的切點,能夠根據運行時的決策(好比方法的參數值)來決定是否應用通知。
通知+切點=切面
引入容許咱們向現有的類添加新方法或屬性
織入是把切面應用到目標對象並建立新的代理對象的過程。切面在指定的鏈接點被織入到目標對象中。在目標對象的生命週期裏有多個點能夠進行織入:
編譯期:切面在目標類編譯時被織入。這種方式須要特殊的編譯器。AspectJ的織入編譯器就是以這種方式織入切面的。
類加載期:切面在目標類加載到JVM時被織入。這種方式須要特殊的類加載器(ClassLoader),它能夠在目標類被引入應用以前加強該目標類的字節碼。AspectJ 5的加載時織入(load-time weaving,LTW)就支持以這種方式織入切面。
運行期:切面在應用運行的某個時刻被織入。通常狀況下,在織入切面時,AOP容器會爲目標對象動態地建立一個代理對象。Spring AOP就是以這種方式織入切面的。
基於代理的經典Spring AOP;
純POJO切面(4.x版本須要XML配置);
@AspectJ註解驅動的切面;
注入式AspectJ切面(適用於Spring各版本)。
前三種都是Spring AOP實現的變體,Spring AOP構建在動態代理基礎之上,所以,Spring對AOP的支持侷限於方法攔截。也就是說,AspectJ纔是王道。
另外在代理類中包裹切面,Spring在運行期把切面織入到Spring管理的bean中。以下圖所示,代理類封裝了目標類,並攔截被通知方法的調用,再把調用轉發給真正的目標bean。當代理攔截到方法調用時,在調用目標bean方法以前,會執行切面邏輯。直到應用須要被代理的bean時,Spring才建立代理對象。 若是使用的是ApplicationContext的話,在ApplicationContext從BeanFactory中加載全部bean的時候,Spring纔會建立被代理的對象。由於Spring運行時才建立代理對象,因此咱們不須要特殊的編譯器來織入Spring AOP的切面。
public interface Performance(){ public void perform(); }
如今來寫一個切點表達式,這個表達式可以設置當perform()方法執行時觸發通知的調用。
execution(* concert.Performance.perform(..))//execution:在方法執行時觸發//*:返回任意類型//concert.Performance:方法所屬類//perform:方法名//(..):使用任意參數
不只如此,還能夠寫的更復雜一點
execution(* concert.Performance.perform(..)&&within(concert.*)) //增長了一個與操做,當concert包下的任意類方法被調用時也會觸發
在切點中選擇bean
execution(*concert.Performance.perform()) and bean('woodstock')//限定bean id爲woodstock
來個完整的切面
@Aspectpublic class Audience{ @Before("execution(**concert.Performance.perform(..))") public void silenceCellPhones(){ System.out.println("Silencing cell phones"); } @Before("execution{** concert.Performance.perform{..}}") public void taskSeats(){ System.out.println("Talking seats"); } @AfterReturning("execution{** concert.Performance.perform{..}}") public void applause(){ System.out.println("CLAP CLAP CLAP!!!"); } @AfterThrowing("execution{** concert.Performance.perform{..}}") public void demanRefund(){ System.out.println("Demanding a refund"); } }
能夠簡化一下
@Aspectpublic class Audience{ //避免頻繁使用切點表達式 @Pointcut("execution(** concert.Performance.perform(..))") public void performance(){} @Before("performance()") public void silenceCellPhones(){ System.out.println("Silencing cell phones"); } @Before("performance()") public void taskSeats(){ System.out.println("Talking seats"); } @AfterReturning("performance()") public void applause(){ System.out.println("CLAP CLAP CLAP!!!"); } @AfterThrowing("performance()") public void demanRefund(){ System.out.println("Demanding a refund"); } }
AOP配置元素 | 用途 |
---|---|
<aop:advisor> | 定義AOP通知器 |
<aop:after> | 定義AOP後置通知(無論被通知的方法是否執行成功) |
<aop:after-returning> | 定義AOP返回通知 |
<aop:after-throwing> | 定義AOP異常通知 |
<aop:around> | 定義AOP環繞通知 |
<aop:aspect> | 定義一個切面 |
<aop:aspectj-autoproxy> | 啓用@AspectJ註解驅動的切面 |
<aop:before> | 定義一個AOP前置通知 |
<aop:config> | 頂層的AOP配置元素。大多數的<aop:*>元素必須包含在<aop:config>元素內 |
<aop:declare-parents> | 以透明的方式爲被通知的對象引入額外的接口 |
<aop:pointcut> | 定義一個切點 |
來個栗子
public class Audience{ public void silenceCellPhones(){ System.out.println("Silencing cell phones"); } public void taskSeats(){ System.out.println("Talking seats"); } public void applause(){ System.out.println("CLAP CLAP CLAP!!!"); } public void demandRefund(){ System.out.println("Demanding a refund"); } }
經過XML將無註解的Audience聲明爲切面
<aop:config> <aop:aspect ref="audience"> <aop:before pointcut ="execution(** concert.Performance.perform(..))" method="sillenceCellPhones"/> <aop:before pointcut ="execution(** concert.Performance.perform(..))" method="taskSeats"/> <aop:after-returning pointcut ="execution(** concert.Performance.perform(..))" method="applause"/> <aop:After-throwing pointcut ="execution(** concert.Performance.perform(..))" method="demanRefund"/> </aop:aspect></aop:config>
AspectJ關於Spring AOP的AspectJ切點,最重要的一點就是Spring僅支持AspectJ切點指示器(pointcut designator)的一個子集。讓咱們回顧下,Spring是基於代理的,而某些切點表達式是與基於代理的AOP無關的。下表列出了Spring AOP所支持的AspectJ切點指示器。
Spring藉助AspectJ的切點表達式語言來定義Spring切面
AspectJ指示器 | 描 述 |
---|---|
arg() | 限制鏈接點匹配參數爲指定類型的執行方法 |
@args() | 限制鏈接點匹配參數由指定註解標註的執行方法 |
execution() | 用於匹配是鏈接點的執行方法 |
this() | 限制鏈接點匹配AOP代理的bean引用爲指定類型的類 |
target | 限制鏈接點匹配目標對象爲指定類型的類 |
@target() | 限制鏈接點匹配特定的執行對象,這些對象對應的類要具備指定類型的註解 |
within() | 限制鏈接點匹配指定的類型 |
@within() | 限制鏈接點匹配指定註解所標註的類型(當使用Spring AOP時,方法定義在由指定的註解所標註的類裏) |
@annotation | 限定匹配帶有指定註解的鏈接點 |
因爲Spring特殊的依賴注入技巧,致使Bean之間沒有耦合度。
可是Bean有時須要使用spring容器自己的資源,這時你的Bean必須意識到Spring容器的存在。因此得使用Spring Aware,下面來看看Spring Aware提供的接口
BeanNameAware | 得到到容器中Bean的名稱 |
---|---|
BeanFactory | 得到當前的bean factory,這樣能夠調用容器的服務 |
ApplicationContextAware* | 當前application context,這樣能夠調用容器的服務 |
MessageSourceAware | 得到Message source |
ApplicationEventPublisherAware | 應用時間發佈器,能夠發佈時間, |
ResourceLoaderAware | 得到資源加載器,能夠得到外部資源文件 |
這樣能夠實現多線程和併發編程。經過@EnableAsync開啓對異步任務的支持,並經過實際執行的Bean的方法始中使用@Async註解來聲明其是一個異步任務
首先經過在配置類註解@EnableScheduling來開啓對計劃任務的支持,而後在要執行計劃任務的方法上註解@Scheduled,聲明這是一個計劃任務
根據知足某一個特定條件建立一個特定的Bean。
元註解就是能夠註解到別的註解上的註解,被註解的註解稱之爲組合註解,組合註解具有註解其上的元註解的功能。
經過觀察這些@Enable*註解的源碼,咱們發現全部的註解都有一個@Import註解,@Import是用來導入配置類的,這也就意外着這些自動開啓的實現實際上是導入了一些自動配置的Bean。這些導入配置的方式主要範圍如下三種類型:
第一類:直接導入配置類
第二類:依據條件選擇配置類
第三類:動態註冊Bean
簡單的分析一下Spring。
Spring 框架中的核心組件只有三個:Core、Context 和 Bean。它們構建起了整個 Spring 的骨骼架構。沒有它們就不可能有 AOP、Web 等上層的特×××。下面也將主要從這三個組件入手分析 Spring。
用過Spring的同窗都知道Bean在Spring的做用是很是重要的。經過一系列簡單的配置來知足類與類之間的依賴關係——這叫作依賴注入。而依賴注入的關係是在一個叫IOC的容器中進行管理。
咱們說到Spring 框架中的核心組件只有三個:Core、Context 和 Bean。那麼Core和Context是如何協做的呢?
咱們知道 Bean 包裝的是 Object,而 Object 必然有數據,如何給這些數據提供生存環境就是 Context 要解決的問題,對 Context 來講他就是要發現每一個 Bean 之間的關係,爲它們創建這種關係而且要維護好這種關係。因此 Context 就是一個 Bean 關係的集合,這個關係集合又叫 Ioc 容器 ,一旦創建起這個 Ioc 容器後 Spring 就能夠爲你工做了。那 Core 組件又有什麼用武之地呢?其實 Core 就是發現、創建和維護每一個 Bean 之間的關係所須要的一些列的工具。
前面已經說明了 Bean 組件對 Spring 的重要性,下面看看 Bean 這個組件式怎麼設計的。Bean 組件在 Spring 的 org.springframework.beans 包下。這個包下的全部類主要解決了三件事:Bean 的定義、Bean 的建立以及對 Bean 的解析。對 Spring 的使用者來講惟一須要關心的就是 Bean 的建立,其餘兩個由 Spring 在內部幫你完成了,對你來講是透明的。
Spring Bean 的建立時典型的工廠模式,他的頂級接口是 BeanFactory,下圖是這個工廠的繼承層次關係:
BeanFactory 有三個子類:ListableBeanFactory、HierarchicalBeanFactory 和 AutowireCapableBeanFactory。可是從上圖中咱們能夠發現最終的默認實現類是 DefaultListableBeanFactory,他實現了全部的接口。那爲什麼要定義這麼多層次的接口呢?查閱這些接口的源碼和說明發現,每一個接口都有他使用的場合,它主要是爲了區分在 Spring 內部在操做過程當中對象的傳遞和轉化過程當中,對對象的數據訪問所作的限制。例如 ListableBeanFactory 接口表示這些 Bean 是可列表的,而 HierarchicalBeanFactory 表示的是這些 Bean 是有繼承關係的,也就是每一個 Bean 有可能有父 Bean。AutowireCapableBeanFactory 接口定義 Bean 的自動裝配規則。這四個接口共同定義了 Bean 的集合、Bean 之間的關係、以及 Bean 行爲。
ApplicationContext 是 Context 的頂級父類,他除了能標識一個應用環境的基本信息外,他還繼承了五個接口,這五個接口主要是擴展了 Context 的功能。下面是 Context 的類結構圖:
從上圖中能夠看出 ApplicationContext
繼承了 BeanFactory,這也說明了 Spring 容器中運行的主體對象是 Bean,另外 ApplicationContext
繼承了 ResourceLoader 接口,使得 ApplicationContext
能夠訪問到任何外部資源,這將在 Core 中詳細說明。
ApplicationContext
的子類主要包含兩個方面:
ConfigurableApplicationContext
表示該 Context 是可修改的,也就是在構建 Context 中用戶能夠動態添加或修改已有的配置信息,它下面又有多個子類,其中最常用的是可更新的 Context,即 AbstractRefreshableApplicationContext
類。
WebApplicationContext
顧名思義,就是爲 web 準備的 Context 他能夠直接訪問到 ServletContext,一般狀況下,這個接口使用的少。
再往下分就是按照構建 Context 的文件類型,接着就是訪問 Context 的方式。這樣一級一級構成了完整的 Context 等級層次。
整體來講 ApplicationContext 必需要完成如下幾件事:
標識一個應用環境
利用 BeanFactory 建立 Bean 對象
保存對象關係表
可以捕獲各類事件
Context 做爲 Spring 的 IOC 容器,基本上整合了 Spring 的大部分功能,或者說是大部分功能的基礎。
Core 組件做爲 Spring 的核心組件,他其中包含了不少的關鍵類,其中一個重要組成部分就是定義了資源的訪問方式。這種把全部資源都抽象成一個接口的方式很值得在之後的設計中拿來學習。下面就重要看一下這個部分在 Spring 的做用。
從上圖能夠看出 Resource 接口封裝了各類可能的資源類型,也就是對使用者來講屏蔽了文件類型的不一樣。對資源的提供者來講,如何把資源包裝起來交給其餘人用這也是一個問題,咱們看到 Resource 接口繼承了 InputStreamSource
接口,這個接口中有個 getInputStream 方法,返回的是 InputStream 類。這樣全部的資源都被能夠經過 InputStream 這個類來獲取,因此也屏蔽了資源的提供者。另外還有一個問題就是加載資源的問題,也就是資源的加載者要統一,從上圖中能夠看出這個任務是由 ResourceLoader 接口完成,他屏蔽了全部的資源加載者的差別,只須要實現這個接口就能夠加載全部的資源,他的默認實現是 DefaultResourceLoader
。
那麼, Context 和 Resource 是如何創建關係的?
從上圖能夠看出,Context 是把資源的加載、解析和描述工做委託給了 ResourcePatternResolver 類來完成,他至關於一個接頭人,他把資源的加載、解析和資源的定義整合在一塊兒便於其餘組件使用。Core 組件中還有不少相似的方式。
該文章算是之前學習Spring時候的一些筆記整理。若是有任何錯誤或者不解的地方,請留言給我。這是一個令咱們能夠一塊兒學習進步的機會。
Java學習交流QQ羣:58980××× 禁止閒聊,非喜勿進!