BeanFactory
BeanFactory是Spring框架最核心的接口,它提供IOC的配置機(jī)制,但是它無(wú)法支持spring的許多插件,如AOP功能、Web應(yīng)用等。 XMLBeanFactory是BeanFactory典型實(shí)現(xiàn)。
注意:
初始化BeanFactory時(shí),必須提供日志框架,在類路徑下提供Log4J配置文件。
BeanFactoroy采用的是延遲加載形式來(lái)注入Bean的,即只有在使用到某個(gè)Bean時(shí)(調(diào)用getBean()),才對(duì)該Bean進(jìn)行加載實(shí)例化,這樣,我們就不能發(fā)現(xiàn)一些存在的Spring的配置問(wèn)題。
ApplicationContext
ApplicationContext接口由BeanFactory接口派生而來(lái),因而提供BeanFactory所有的功能。ApplicationContext以一種更向面向框架的方式工作以及對(duì)上下文進(jìn)行分層和實(shí)現(xiàn)繼承,ApplicationContext包還提供了以下的功能:
? MessageSource, 提供國(guó)際化的消息訪問(wèn)
? 資源訪問(wèn),如URL和文件
? 事件傳播
? 載入多個(gè)(有繼承關(guān)系)上下文 ,使得每一個(gè)上下文都專注于一個(gè)特定的層次,比如應(yīng)用的web層
利用MessageSource進(jìn)行國(guó)際化
BeanFactory是不支持國(guó)際化功能的,因?yàn)锽eanFactory沒(méi)有擴(kuò)展Spring中MessageResource接口。相反,由于ApplicationContext擴(kuò)展了MessageResource接口,因而具有消息處理的能力(i18N),具體spring如何使用國(guó)際化,以后章節(jié)會(huì)詳細(xì)描述。強(qiáng)大的事件機(jī)制(Event)
基本上牽涉到事件(Event)方面的設(shè)計(jì),就離不開(kāi)觀察者模式。
ApplicationContext的事件機(jī)制主要通過(guò)ApplicationEvent和ApplicationListener這兩個(gè)接口來(lái)提供的,和java swing中的事件機(jī)制一樣。即當(dāng)ApplicationContext中發(fā)布一個(gè)事件的時(shí),所有擴(kuò)展了ApplicationListener的Bean都將會(huì)接受到這個(gè)事件,并進(jìn)行相應(yīng)的處理。
Spring提供了部分內(nèi)置事件,主要有以下幾種:
ContextRefreshedEvent :ApplicationContext發(fā)送該事件時(shí),表示該容器中所有的Bean都已經(jīng)被裝載完成,此ApplicationContext已就緒可用
ContextStartedEvent:生命周期 beans的啟動(dòng)信號(hào)
ContextStoppedEvent: 生命周期 beans的停止信號(hào)
ContextClosedEvent:ApplicationContext關(guān)閉事件,則context不能刷新和重啟,從而所有的singleton bean全部銷毀(因?yàn)閟ingleton bean是存在容器緩存中的)
雖然,spring提供了許多內(nèi)置事件,但用戶也可根據(jù)自己需要來(lái)擴(kuò)展spriong中的事物。注意,要擴(kuò)展的事件都要實(shí)現(xiàn)ApplicationEvent接口。
底層資源的訪問(wèn)
ApplicationContext擴(kuò)展了ResourceLoader(資源加載器)接口,從而可以用來(lái)加載多個(gè)Resource,而BeanFactory是沒(méi)有擴(kuò)展ResourceLoader對(duì)Web應(yīng)用的支持
與BeanFactory通常以編程的方式被創(chuàng)建不同的是,ApplicationContext能以聲明的方式創(chuàng)建,如使用ContextLoader。當(dāng)然你也可以使用ApplicationContext的實(shí)現(xiàn)之一來(lái)以編程的方式創(chuàng)建ApplicationContext實(shí)例 。
注意:ApplicationContext則相反,它是在容器啟動(dòng)時(shí),一次性創(chuàng)建了所有的Bean。這樣,在容器啟動(dòng)時(shí),我們就可以發(fā)現(xiàn)Spring中存在的配置錯(cuò)誤。
ApplicationContext繼承了ResourceLoader接口使得ApplicationContext可以訪問(wèn)到任何外部資源。
WebApplicationContext
WebApplicationContext是專門為Web應(yīng)用準(zhǔn)備的,它允許從相對(duì)于Web的根目錄的路徑裝載配置文件來(lái)完成初始化。從WebApplicationContext中可以獲得ServletContext引用,整個(gè)Web應(yīng)用上下文對(duì)象講作為屬性放置在ServletContext中,以便Web引用環(huán)境可以訪問(wèn)Spring的上下文。
WebApplicationContext初始化
需要擁有Web容器的前提下才可以啟動(dòng),啟動(dòng)方式有在web.xml下配置自啟動(dòng)的servlet(ContextLoaderServlet)或者定義Web容器監(jiān)聽(tīng)器(ContextLoaderListener);
。
這個(gè)listener需要檢查contextConfigLocation參數(shù)。如果不存在的話,它將默認(rèn)使用/WEB-INF/applicationContext.xml。如果它存在,它就會(huì)用預(yù)先定義的分隔符(逗號(hào),分號(hào)和空格)分開(kāi)分割字符串,并將這些值作為應(yīng)用上下文將要搜索的位置。ContextLoaderServlet可以用來(lái)替換ContextLoaderListener。這個(gè)servlet像listener那樣使用contextConfigLocation參數(shù)。
Spring和SpringMVC父子容器
(1) 通過(guò)HierarchicalBeanFactory接口,Spring的IoC容器可以建立父子層級(jí)關(guān)聯(lián)的容器體系,子容器可以訪問(wèn)父容器中的Bean,但父容器不能訪問(wèn)子容器的Bean。在容器內(nèi),Bean的id必須是唯一的,但子容器可以擁有一個(gè)和父容器id相同的Bean。父子容器層級(jí)體系增強(qiáng)了Spring容器架構(gòu)的擴(kuò)展性和靈活性,因?yàn)榈谌娇梢酝ㄟ^(guò)編程的方式,為一個(gè)已經(jīng)存在的容器添加一個(gè)或多個(gè)特殊用途的子容器,以提供一些額外的功能。
(2)Spring使用父子容器實(shí)現(xiàn)了很多功能,比如在Spring MVC中,展現(xiàn)層Bean位于一個(gè)子容器中,而業(yè)務(wù)層和持久層的Bean位于父容器中。這樣,展現(xiàn)層Bean就可以引用業(yè)務(wù)層和持久層的Bean,而業(yè)務(wù)層和持久層的Bean則看不到展現(xiàn)層的Bean。
代碼解讀
配置Spring容器
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath*:applicationContext-*.xml</param-value>
</context-param>
//ContextLoaderListener中initWebApplicationContext()方法源碼片段
//將Spring容器設(shè)置到servletContext
servletContext.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, this.context);
//FrameWorkServlet中initWebApplicationContext()方法源碼片段
ConfigurableWebApplicationContext cwac = (ConfigurableWebApplicationContext) wac;
cwac.setParent(rootContext);
//獲取rootContext
public static WebApplicationContext getWebApplicationContext(ServletContext sc) {
//從servletContext中獲得Spring容器
return getWebApplicationContext(sc, WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE);
}