<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
隨著SpringBoot的普及,Spring的使用也越來越廣,在某些場景下,我們無法通過註解或設定的形式直接獲取到某個Bean。比如,在某一些工具類、設計模式實現中需要使用到Spring容器管理的Bean,此時就需要直接獲取到對應的Bean。
本文為大家整理彙總了常見的獲取Bean的方式,並提供一些優劣分析,方便大家在使用到時有更好的選擇。同時,也會為大家適當的普及和拓展一些相關知識。
在Spring中,Bean的範例化、定位、設定應用程式中的物件及建立物件間的依賴關係,都是在IoC容器中進行的。因此,要在Spring中獲取Bean,本質上就是從IoC容器當中獲取Bean。
在Spring中,BeanFactory是IoC容器的實際代表者,該介面提供了IoC容器最基本功能。同時,Spring還提供了另外一種型別的容器:ApplicationContext容器。
ApplicationContext容器包括BeanFactory容器的所有功能(BeanFactory的子介面),提供了更多面嚮應用的功能,它提供了國際化支援和框架事件體系,更易於建立實際應用。
一般情況,我們稱BeanFactory為IoC容器,稱ApplicationContext為應用上下文。但有時為了方便,也將ApplicationContext稱為Spring容器。
通常不建議使用BeanFactory,但BeanFactory 仍然可以用於輕量級的應用程式,如移動裝置或基於applet的應用程式,其中它的資料量和速度是顯著。
BeanFactory是Spring框架的基礎設施,面向Spring本身。ApplicationContext則面向使用Spring框架的開發者,幾乎所有的應用場合都可以直接使用ApplicationContext,而非底層的BeanFactory。
另外,ApplicationContext的初始化和BeanFactory有一個重大的區別:
BeanFactory在初始化容器時,並未範例化Bean,直到第一次存取某個Bean時才範例目標Bean。這樣,我們就不能發現一些存在的Spring的設定問題。如果Bean的某一個屬性沒有注入,BeanFacotry載入後,直至第一次使用呼叫getBean方法才會丟擲異常。
而ApplicationContext則在初始化應用上下文時就範例化所有單範例的Bean,相對應的,ApplicationContext的初始化時間會比BeanFactory長一些。
瞭解了上述的基本理論知識之後,我們就可以嘗試從IoC容器當中獲取Bean物件了。
通過BeanFactory來獲取Bean。基於xml組態檔的時代,可以通過如下方式獲得BeanFactory,再通過BeanFactory來獲得對應的Bean。
BeanFactory beanFactory = new XmlBeanFactory(new ClassPathResource("applicationContext.xml")); UserInfo userInfo = (UserInfo) beanFactory.getBean("userInfo");
有一定程式設計年齡的程式設計師,應該對此還有一些印象。這種寫法估計也只會出現在古老的專案當中。鑑於xml形式組態檔已經被基於註解形式所替代,同時XmlBeanFactory也被標註為廢棄。此種方式不推薦使用。
其實,不推薦的理由還有一個,在上面已經提到,儘量不要使用BeanFactory,而應該使用ApplicationContext。
在上面的方式中,XmlBeanFactory已經被廢棄,但可以通過其他方式來獲得BeanFactory,然後再從BeanFactory中獲得指定的Bean。獲取BeanFactory範例最簡單的方式就是實現BeanFactoryAware介面。
BeanFactoryAware介面原始碼:
public interface BeanFactoryAware extends Aware { /** * 初始化回撥方法,Spring會自動將BeanFactory注入進去,接收之後即可使用BeanFactory */ void setBeanFactory(BeanFactory beanFactory) throws BeansException; }
BeanFactoryAware屬於org.springframework.beans.factory.Aware根標記介面,使用setter注入來在應用程式上下文啟動期間獲取物件。Aware介面是回撥,監聽器和觀察者設計模式的混合,它表示Bean有資格通過回撥方式被Spring容器通知。
這裡提供一個完整的工具類:
@Component public class BeanFactoryHelper implements BeanFactoryAware { private static BeanFactory beanFactory; /** * 重寫 BeanFactoryAware 介面的方法 * @param beanFactory :引數賦值給本地屬性之後即可使用 BeanFactory * @throws BeansException BeansException */ @Override public void setBeanFactory(BeanFactory beanFactory) throws BeansException { BeanFactoryHelper.beanFactory = beanFactory; } /** * 根據名稱獲取容器中的物件範例 * @param beanName :注入的範例必須已經存在容器中,否則拋異常:NoSuchBeanDefinitionException * @return Object */ public static Object getBean(String beanName) { return beanFactory.getBean(beanName); } /** * 根據 class 獲取容器中的物件範例 * @param requiredType :被注入的必須已經存在容器中,否則拋異常:NoSuchBeanDefinitionException * @param <T> Class * @return 物件 */ public static <T> T getBean(Class<T> requiredType) { return beanFactory.getBean(requiredType); } /** * 判斷 spring 容器中是否包含指定名稱的物件 * @param beanName bean名稱 * @return 是否存在 */ public static boolean containsBean(String beanName) { return beanFactory.containsBean(beanName); } //其它需求皆可參考 BeanFactory 介面和它的實現類 }
在上述工具類中,便是基於BeanFactoryAware的特性,獲得了BeanFactory,然後再通過BeanFactory來獲得指定的Bean。
該方案滿足了獲取Bean的基本需求,但同時具有使用BeanFactory的缺點。根據前文介紹的BeanFactory特性,可酌情使用。
上面提供了兩種基於BeanFactory容器獲得Bean的方式,下面則通過ApplicationContext來獲取容器中的Bean,不同的是獲取ApplicationContext的方式的區別。
在專案啟動時先獲取ApplicationContext物件,然後將其儲存在一個地方,以便後續用到時進行使用。
這裡提供兩種場景的獲取:
基於xml設定bean的形式,適用於比較古老的專案,已經很少使用了;
基於SpringBoot啟動時獲取ApplicationContext物件;
基於xml的形式實現:
// 其中applicationContext.xml 為設定容器的xml,不過現在一般很少使用了 ApplicationContext ac = new FileSystemXmlApplicationContext("applicationContext.xml");
這裡等於直接初始化容器,並且獲得容器的參照。這種方式適用於採用Spring框架的獨立應用程式,需要程式通過組態檔手工初始化Spring的情況。目前大多數Spring專案已經不再採用xml設定,很少使用了。
基於SpringBoot啟動實現:
@SpringBootApplication public class ExampleApplication { public static void main(String[] args) { // 啟動時,儲存上下文,並儲存為靜態 ConfigurableApplicationContext ac = SpringApplication.run(ExampleApplication.class, args); SpringContextUtil.setAc(ac); } }
對應的SpringContextUtil類如下:
public class SpringContextUtil1 { private static ApplicationContext ac; public static <T> T getBean(String beanName, Class<T> clazz) { T bean = ac.getBean(beanName, clazz); return bean; } public static void setAc(ApplicationContext applicationContext){ ac = applicationContext; } }
兩種方式都是在啟動Spring專案時,直接獲取到ApplicationContext的參照,然後將其儲存到工具類當中。在使用時,則從工具類中獲取ApplicationContext容器,進而從中獲得Bean物件。
此種方式依舊是先獲得ApplicationContext容器,然後從中獲取Bean物件,只不過是基於繼承ApplicationObjectSupport類實現的。
具體實現程式碼:
@Component public class SpringContextUtil extends ApplicationObjectSupport { public <T> T getBean(Class<T> clazz) { ApplicationContext ac = getApplicationContext(); if(ac == null){ return null; } return ac.getBean(clazz); } }
注意,這裡的SpringContextUtil類需要範例化。
WebApplicationObjectSupport是ApplicationObjectSupport的一個實現類,提供了Web相關的支援。實現原理與ApplicationObjectSupport一樣。
具體實現程式碼如下:
@Component public class SpringContextUtil extends WebApplicationObjectSupport { public <T> T getBean(Class<T> clazz) { ApplicationContext ac = getApplicationContext(); if(ac == null){ return null; } return ac.getBean(clazz); } }
對照基於ApplicationObjectSupport的實現,除了繼承物件不同外,沒有其他區別,都是基於getApplicationContext方法來獲取。
Spring提供了工具類WebApplicationContextUtils,通過該類可獲取WebApplicationContext物件。
具體實現程式碼如下:
public class SpringContextUtil2 { public static <T> T getBean(ServletContext request, String name, Class<T> clazz){ WebApplicationContext webApplicationContext = WebApplicationContextUtils.getRequiredWebApplicationContext(request); // 或者 WebApplicationContext webApplicationContext1 = WebApplicationContextUtils.getWebApplicationContext(request); // webApplicationContext1.getBean(name, clazz) T bean = webApplicationContext.getBean(name, clazz); return bean; } }
這個方法很常見於SpringMVC構建的Web專案中,適用於Web專案的B/S結構。
通過實現ApplicationContextAware介面,在Spring容器啟動時將ApplicationContext注入進去,從而獲取ApplicationContext物件,這種方法也是常見的獲取Bean的一種方式,推薦使用。
具體實現程式碼如下:
@Component public class SpringContextUtil3 implements ApplicationContextAware { private static ApplicationContext ac; @Override public void setApplicationContext(ApplicationContext applicationContext) throws BeansException { ac = applicationContext; } public static <T> T getBean(Class<T> clazz) { T bean = ac.getBean(clazz); return bean; } }
這種方式與前面通過BeanFactoryAware獲得BeanFactory的思路一致。
使用ContextLoader提供的getCurrentWebApplicationContext方法,也是常用的獲取WebApplicationContext的一種方法。
具體實現程式碼如下:
WebApplicationContext wac = ContextLoader.getCurrentWebApplicationContext(); wac.getBean(beanID);
該方法常見於SpringMVC實現的Web專案中。該方式是一種不依賴於Servlet,不需要注入的方式。但是需要注意一點,在伺服器啟動時和Spring容器初始化時,不能通過該方法獲取Spring容器。
Spring工具類,方便在非Spring管理環境中獲取Bean。
@Component public final class SpringUtils implements BeanFactoryPostProcessor{ /** Spring應用上下文環境 */ private static ConfigurableListableBeanFactory beanFactory; @Override public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException{ SpringUtilsS.beanFactory = beanFactory; } /** * 獲取物件 * * @param name * @return Object 一個以所給名字註冊的bean的範例 * @throws BeansException * */ @SuppressWarnings("unchecked") public static <T> T getBean(String name) throws BeansException{ return (T) beanFactory.getBean(name); } /** * 獲取型別為requiredType的物件 * * @param clz * @return * @throws BeansException * */ public static <T> T getBean(Class<T> clz) throws BeansException{ T result = (T) beanFactory.getBean(clz); return result; } /** * 如果BeanFactory包含一個與所給名稱匹配的bean定義,則返回true * * @param name * @return boolean */ public static boolean containsBean(String name){ return beanFactory.containsBean(name); } /** * 判斷以給定名字註冊的bean定義是一個singleton還是一個prototype。 如果與給定名字相應的bean定義沒有被找到,將會丟擲一個異常(NoSuchBeanDefinitionException) * * @param name * @return boolean * @throws NoSuchBeanDefinitionException * */ public static boolean isSingleton(String name) throws NoSuchBeanDefinitionException{ return beanFactory.isSingleton(name); } /** * @param name * @return Class 註冊物件的型別 * @throws NoSuchBeanDefinitionException * */ public static Class<?> getType(String name) throws NoSuchBeanDefinitionException{ return beanFactory.getType(name); } /** * 如果給定的bean名字在bean定義中有別名,則返回這些別名 * * @param name * @return * @throws NoSuchBeanDefinitionException * */ public static String[] getAliases(String name) throws NoSuchBeanDefinitionException{ return beanFactory.getAliases(name); } /** * 獲取aop代理物件 * * @param invoker * @return */ @SuppressWarnings("unchecked") public static <T> T getAopProxy(T invoker){ return (T) AopContext.currentProxy(); } }
其中ConfigurableListableBeanFactory介面,也屬於BeanFactory的子介面。
在本文中介紹了9種從Spring容器中獲取Bean的方法,雖然每種方式實現各有不同,但從本質上來講,無非就是通過BeanFactory或ApplicationContext獲取Bean,只不過獲取BeanFactory或ApplicationContext容器的方式不同而已。
那麼,你是否意識到,學習一項技術或一個實現方式,只要把握住它的根本,無論形式如何變化,都萬變不離其宗。而這裡“宗”就是IoC容器。
到此這篇關於淺析Spring獲取Bean的九種方法詳解的文章就介紹到這了,更多相關Spring獲取Bean內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45