
1、在初始化時保存ApplicationContext對象
適用于Spring框架的獨立應用程序,須要程序通過配置文件初始化Spring。
applicationContext.xml配置:
代碼:
@Test publicvoidtest(){ ApplicationContextapplicationContext=newClassPathXmlApplicationContext("applicationContext.xml"); //ApplicationContextapplicationContext=newFileSystemXmlApplicationContext("applicationContext.xml"); Testtest=(Test)applicationContext.getBean("test"); System.out.println(test); }
2、通過Spring提供的工具類獲取ApplicationContext對象
適合于Spring框架的B/S系統,通過ServletContext對象獲取ApplicationContext對象。然后在通過它獲取須要的類實例。以下兩個工具方式的差別是,前者在獲取失敗時拋出異常。后者返回null。
ApplicationContextac1=WebApplicationContextUtils.getRequiredWebApplicationContext(ServletContextsc); ApplicationContextac2=WebApplicationContextUtils.getWebApplicationContext(ServletContextsc); ac1.getBean("beanId"); ac2.getBean("beanId");
3、實現接口ApplicationContextAware(推薦)
實現該接口的setApplicationContext(ApplicationContext context)方法,并保存ApplicationContext 對象。Spring初始化時,掃描到該類,就會通過該方法將ApplicationContext對象注入。然后在代碼中就可以獲取spring容器bean了。
例如:
User bean = SpringUtils.getBean(“user”);
@Component publicclassSpringUtilsimplementsApplicationContextAware{ privatestaticApplicationContextapplicationContext; @Override publicvoidsetApplicationContext(ApplicationContextapplicationContext)throwsBeansException{ SpringUtils.applicationContext=applicationContext; } publicstaticTgetBean(StringbeanName){ if(applicationContext.containsBean(beanName)){ return(T)applicationContext.getBean(beanName); }else{ returnnull; } } publicstatic Map getBeansOfType(Class baseType){ returnapplicationContext.getBeansOfType(baseType); } }
4、繼承自抽象類ApplicationObjectSupport
調用父類的getApplicationContext()方法,獲取Spring容器對象。
@Service
publicclassSpringContextHelperextendsApplicationObjectSupport{
publicObjectgetBean(StringbeanName){
returngetApplicationContext().getBean(beanName);
}
}
5、繼承自抽象類WebApplicationObjectSupport
調用getWebApplicationContext()獲取WebApplicationContext
@Service
publicclassSpringContextHelperextendsWebApplicationObjectSupport{
publicObjectgetBean(StringbeanName){
returngetApplicationContext().getBean(beanName);
}
}
6、使用BeanFactory直接獲取(不推薦)
使用BeanFactory從工廠中直接獲取Bean實例,但是XmlBeanFactory類已經廢棄,因此不建議使用。
@Test
publicvoidtest(){
BeanFactorybeanFactory=newXmlBeanFactory(newClassPathResource("applicationContext.xml"));
Testtest=(Test)beanFactory.getBean("test");
System.out.println(test);
}
7、使用ContextLoader提供的getCurrentWebApplicationContext方法
@Test
publicvoidtest(){
MockServletContextsc=newMockServletContext("");
sc.addInitParameter(ContextLoader.CONFIG_LOCATION_PARAM,"/applicationContext.xml");
ServletContextListenerlistener=newContextLoaderListener();
ServletContextEventevent=newServletContextEvent(sc);
listener.contextInitialized(event);
WebApplicationContextwac=ContextLoader.getCurrentWebApplicationContext();
Testtest=(Test)wac.getBean("test");
System.out.println(test);
}
8、實現接口BeanFactoryPostProcessor
spring工具類 方便在非spring管理環境中獲取bean
@Component
publicfinalclassSpringUtilsSimplementsBeanFactoryPostProcessor
{
/**Spring應用上下文環境*/
privatestaticConfigurableListableBeanFactorybeanFactory;
@Override
publicvoidpostProcessBeanFactory(ConfigurableListableBeanFactorybeanFactory)throwsBeansException
{
SpringUtilsS.beanFactory=beanFactory;
}
/**
*獲取對象
*
*@paramname
*@returnObject一個以所給名字注冊的bean的實例
*@throwsBeansException
*
*/
@SuppressWarnings("unchecked")
publicstaticTgetBean(Stringname)throwsBeansException
{
return(T)beanFactory.getBean(name);
}
/**
*獲取類型為requiredType的對象
*
*@paramclz
*@return
*@throwsBeansException
*
*/
publicstaticTgetBean(Classclz)throwsBeansException
{
Tresult=(T)beanFactory.getBean(clz);
returnresult;
}
/**
*如果BeanFactory包含一個與所給名稱匹配的bean定義,則返回true
*
*@paramname
*@returnboolean
*/
publicstaticbooleancontainsBean(Stringname)
{
returnbeanFactory.containsBean(name);
}
/**
*判斷以給定名字注冊的bean定義是一個singleton還是一個prototype。如果與給定名字相應的bean定義沒有被找到,將會拋出一個異常(NoSuchBeanDefinitionException)
*
*@paramname
*@returnboolean
*@throwsNoSuchBeanDefinitionException
*
*/
publicstaticbooleanisSingleton(Stringname)throwsNoSuchBeanDefinitionException
{
returnbeanFactory.isSingleton(name);
}
/**
*@paramname
*@returnClass注冊對象的類型
*@throwsNoSuchBeanDefinitionException
*
*/
publicstaticClass>getType(Stringname)throwsNoSuchBeanDefinitionException
{
returnbeanFactory.getType(name);
}
/**
*如果給定的bean名字在bean定義中有別名,則返回這些別名
*
*@paramname
*@return
*@throwsNoSuchBeanDefinitionException
*
*/
publicstaticString[]getAliases(Stringname)throwsNoSuchBeanDefinitionException
{
returnbeanFactory.getAliases(name);
}
/**
*獲取aop代理對象
*
*@paraminvoker
*@return
*/
@SuppressWarnings("unchecked")
publicstaticTgetAopProxy(Tinvoker)
{
return(T)AopContext.currentProxy();
}
}
擴展
BeanFactory和ApplicationContext是Spring的兩大核心接口,都可以當做Spring的容器。其中ApplicationContext是BeanFactory的子接口。
BeanFactory
(1)、是Spring里面最底層的接口(最原始的接口),包含了各種Bean的定義,讀取bean配置文檔,管理bean的加載、實例化,控制bean的生命周期,維護bean之間的依賴關系。
(2)、采用的是延遲加載形式來注入Bean的,即只有在使用到某個Bean時(調用getBean()),才對該Bean進行加載實例化。這樣,我們就不能發現一些存在的Spring的配置問題。如果Bean的某一個屬性沒有注入,BeanFacotry加載后,直至第一次使用調用getBean方法才會拋出異常。
(3)BeanFactory通常以編程的方式被創建。
(4)BeanFactory和ApplicationContext都支持BeanPostProcessor、BeanFactoryPostProcessor的使用,但兩者之間的區別是:BeanFactory需要手動注冊,而ApplicationContext則是自動注冊。
(5) 占用內存小。
ApplicationContext
1、ApplicationContext接口作為BeanFactory的派生,除了提供BeanFactory所具有的功能外,還提供了更完整的框架功能:
繼承MessageSource,因此支持國際化。
統一的資源文件訪問方式。
提供在監聽器中注冊bean的事件。
同時加載多個配置文件。
載入多個(有繼承關系)上下文 ,使得每一個上下文都專注于一個特定的層次,比如應用的web層。
2、ApplicationContext,它是在容器啟動時,一次性創建了所有的Bean。這樣,在容器啟動時,我們就可以發現Spring中存在的配置錯誤,這樣有利于檢查所依賴屬性是否注入。ApplicationContext啟動后預載入所有的單實例Bean,通過預載入單實例bean ,確保當你需要的時候,你就不用等待,因為它們已經創建好了。
3、ApplicationContext 占用內存空間大,當程序的配置bean特別多時,程序啟動慢。
4、ApplicationContext 能以編程式方式創建,還能能以聲明的方式創建,如使用ContextLoader。
審核編輯:湯梓紅
-
接口
+關注
關注
33文章
9519瀏覽量
157019 -
框架
+關注
關注
0文章
404瀏覽量
18421 -
spring
+關注
關注
0文章
341瀏覽量
15935
原文標題:Spring中獲取bean的八種方式,你get了幾種?
文章出處:【微信號:AndroidPush,微信公眾號:Android編程精選】歡迎添加關注!文章轉載請注明出處。
發布評論請先 登錄
java spring教程
什么是java spring
spring實例
三大框架之Spring
「Spring認證」Spring Hello World 項目示例
Spring應用 1 springXML配置說明
解析加載及實例化Bean的順序(零配置)
「Spring認證」Spring IoC 容器
Spring中Bean的生命周期是怎樣的?
Spring Dependency Inject與Bean Scops注解
Spring依賴注入Bean類型的8種情況
Spring容器原始Bean是如何創建的?Spring源碼中方法的執行順序
Spring中獲取bean的八種方式
評論