Spring IoC容器常見獲取Bean的方式匯總示例解析
1 前言
隨著SpringBoot的普及,Spring的使用也越來越廣,在某些場(chǎng)景下,我們無法通過注解或配置的形式直接獲取到某個(gè)Bean。比如,在某一些工具類、設(shè)計(jì)模式實(shí)現(xiàn)中需要使用到Spring容器管理的Bean,此時(shí)就需要直接獲取到對(duì)應(yīng)的Bean。
本文為大家整理匯總了常見的獲取Bean的方式,并提供一些優(yōu)劣分析,方便大家在使用到時(shí)有更好的選擇。同時(shí),也會(huì)為大家適當(dāng)?shù)钠占昂屯卣挂恍┫嚓P(guān)知識(shí)。
2 Spring的IoC容器
在Spring中,Bean的實(shí)例化、定位、配置應(yīng)用程序中的對(duì)象及建立對(duì)象間的依賴關(guān)系,都是在IoC容器中進(jìn)行的。因此,要在Spring中獲取Bean,本質(zhì)上就是從IoC容器當(dāng)中獲取Bean。
在Spring中,BeanFactory是IoC容器的實(shí)際代表者,該接口提供了IoC容器最基本功能。同時(shí),Spring還提供了另外一種類型的容器:ApplicationContext容器。
ApplicationContext容器包括BeanFactory容器的所有功能(BeanFactory的子接口),提供了更多面向應(yīng)用的功能,它提供了國(guó)際化支持和框架事件體系,更易于創(chuàng)建實(shí)際應(yīng)用。
一般情況,我們稱BeanFactory為IoC容器,稱ApplicationContext為應(yīng)用上下文。但有時(shí)為了方便,也將ApplicationContext稱為Spring容器。
通常不建議使用BeanFactory,但BeanFactory 仍然可以用于輕量級(jí)的應(yīng)用程序,如移動(dòng)設(shè)備或基于applet的應(yīng)用程序,其中它的數(shù)據(jù)量和速度是顯著。
BeanFactory與ApplicationContext的區(qū)別
BeanFactory是Spring框架的基礎(chǔ)設(shè)施,面向Spring本身。ApplicationContext則面向使用Spring框架的開發(fā)者,幾乎所有的應(yīng)用場(chǎng)景都可以直接使用ApplicationContext,而非底層的BeanFactory。
另外,ApplicationContext的初始化和BeanFactory有一個(gè)重大的區(qū)別:
BeanFactory在初始化容器時(shí),并未實(shí)例化Bean,直到第一次訪問某個(gè)Bean時(shí)才實(shí)例目標(biāo)Bean。這樣,我們就不能發(fā)現(xiàn)一些存在的Spring的配置問題。如果Bean的某一個(gè)屬性沒有注入,BeanFacotry加載后,直至第一次使用調(diào)用getBean方法才會(huì)拋出異常。
而ApplicationContext則在初始化應(yīng)用上下文時(shí)就實(shí)例化所有單實(shí)例的Bean,相對(duì)應(yīng)的,ApplicationContext的初始化時(shí)間會(huì)比BeanFactory長(zhǎng)一些。
了解了上述的基本理論知識(shí)之后,我們就可以嘗試從IoC容器當(dāng)中獲取Bean對(duì)象了。
3 Bean獲取方式
3.1 方式一:通過BeanFactory獲取
通過BeanFactory來獲取Bean?;趚ml配置文件的時(shí)代,可以通過如下方式獲得BeanFactory,再通過BeanFactory來獲得對(duì)應(yīng)的Bean。
BeanFactory beanFactory = new XmlBeanFactory(new ClassPathResource("applicationContext.xml"));
User user = (User) beanFactory.getBean("user");
有一定編程年齡的程序員,應(yīng)該對(duì)此還有一些印象。這種寫法估計(jì)也只會(huì)出現(xiàn)在古老的項(xiàng)目當(dāng)中。鑒于xml形式配置文件已經(jīng)被基于注解形式所替代,同時(shí)XmlBeanFactory也被標(biāo)注為廢棄。此種方式不推薦使用。
其實(shí),不推薦的理由還有一個(gè),在上面已經(jīng)提到,盡量不要使用BeanFactory,而應(yīng)該使用ApplicationContext。
3.2 方式二 :通過BeanFactoryAware獲取
在上面的方式中,XmlBeanFactory已經(jīng)被廢棄,但可以通過其他方式來獲得BeanFactory,然后再?gòu)腂eanFactory中獲得指定的Bean。獲取BeanFactory實(shí)例最簡(jiǎn)單的方式就是實(shí)現(xiàn)BeanFactoryAware接口。
BeanFactoryAware接口源碼: public interface BeanFactoryAware extends Aware { /** * 初始化回調(diào)方法,Spring會(huì)自動(dòng)將BeanFactory注入進(jìn)去,接收之后即可使用BeanFactory */ void setBeanFactory(BeanFactory beanFactory) throws BeansException; } BeanFactoryAware屬于 org.springframework.beans.factory.Aware根標(biāo)記接口,使用setter注入來在應(yīng)用程序上下文啟動(dòng)期間獲取對(duì)象。Aware接口是回調(diào),監(jiān)聽器和觀察者設(shè)計(jì)模式的混合,它表示Bean有資格通過回調(diào)方式被Spring容器通知。 示例如下: @Component public class BeanFactoryHelper implements BeanFactoryAware { private static BeanFactory beanFactory; /** * 重寫 BeanFactoryAware 接口的方法 * @param beanFactory :參數(shù)賦值給本地屬性之后即可使用 BeanFactory * @throws BeansException BeansException */ @Override public void setBeanFactory(BeanFactory beanFactory) throws BeansException { BeanFactoryHelper.beanFactory = beanFactory; } /** * 根據(jù)名稱獲取容器中的對(duì)象實(shí)例 * @param beanName :注入的實(shí)例必須已經(jīng)存在容器中,否則拋異常:NoSuchBeanDefinitionException * @return Object */ public static Object getBean(String beanName) { return beanFactory.getBean(beanName); } /** * 根據(jù) class 獲取容器中的對(duì)象實(shí)例 * @param requiredType :被注入的必須已經(jīng)存在容器中,否則拋異常:NoSuchBeanDefinitionException * @param <T> Class * @return 對(duì)象 */ public static <T> T getBean(Class<T> requiredType) { return beanFactory.getBean(requiredType); } /** * 判斷 spring 容器中是否包含指定名稱的對(duì)象 * @param beanName bean名稱 * @return 是否存在 */ public static boolean containsBean(String beanName) { return beanFactory.containsBean(beanName); } //其它需求皆可參考 BeanFactory 接口和它的實(shí)現(xiàn)類 }
以上就是Spring IoC容器常見獲取Bean的方式匯總示例解析的詳細(xì)內(nèi)容,更多關(guān)于Spring IoC容器獲取Bean的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
在Java中實(shí)現(xiàn)線程安全的單例模式的常見方式
單例模式是一種常用的軟件設(shè)計(jì)模式,它確保一個(gè)類只有一個(gè)實(shí)例,并提供一個(gè)全局訪問點(diǎn),在多線程環(huán)境下,確保單例模式的線程安全性是非常重要的,因?yàn)槎鄠€(gè)線程可能會(huì)同時(shí)嘗試創(chuàng)建實(shí)例,導(dǎo)致實(shí)例不唯一的問題,本文介紹了在Java中實(shí)現(xiàn)線程安全的單例模式有幾種常見的方式2024-09-09Mybatis-Plus 條件構(gòu)造器 QueryWrapper 的基本用法
這篇文章主要介紹了Mybatis-Plus - 條件構(gòu)造器 QueryWrapper 的使用,通過實(shí)例代碼給大家介紹了查詢示例代碼及實(shí)現(xiàn)需求,代碼簡(jiǎn)單易懂,對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-09-09Java 中HttpURLConnection附件上傳的實(shí)例詳解
這篇文章主要介紹了Java 中HttpURLConnection附件上傳的實(shí)例詳解的相關(guān)資料,希望通過本文大家能掌握這樣的知識(shí)內(nèi)容,需要的朋友可以參考下2017-09-09SpringBoot整合Prometheus如何實(shí)現(xiàn)資源監(jiān)控
本文介紹了如何使用Prometheus監(jiān)控SpringBoot應(yīng)用,Prometheus是一個(gè)開源的監(jiān)控和告警工具,SpringBootActuator提供了監(jiān)控和管理SpringBoot應(yīng)用的工具,通過添加依賴、配置Actuator和Prometheus,可以實(shí)現(xiàn)對(duì)SpringBoot應(yīng)用的實(shí)時(shí)監(jiān)控2024-12-12Idea跑的項(xiàng)目沒問題將程序install成jar包運(yùn)行報(bào)錯(cuò)空指針的問題
這篇文章主要介紹了Idea跑的項(xiàng)目沒問題,將程序install成jar包運(yùn)行報(bào)錯(cuò)空指針的問題,本文通過圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-06-06MyBatis主鍵生成策略中useGeneratedKeys和<selectKey>的區(qū)別
本文主要介紹了MyBatis主鍵生成策略中useGeneratedKeys和<selectKey>的區(qū)別,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2025-01-01Java通過PropertyDescriptor反射調(diào)用set和get方法
這篇文章主要為大家詳細(xì)介紹了Java通過PropertyDescriptor反射調(diào)用set和get方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-03-03