成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Spring容器獲取Bean的九種方式

開發 前端
在本文中介紹了9種從Spring容器中獲取Bean的方法,雖然每種方式實現各有不同,但從本質上來講,無非就是通過BeanFactory或ApplicationContext獲取Bean,只不過獲取BeanFactory或ApplicationContext容器的方式不同而已。

前言

隨著SpringBoot的普及,Spring的使用也越來越廣,在某些場景下,我們無法通過注解或配置的形式直接獲取到某個Bean。比如,在某一些工具類、設計模式實現中需要使用到Spring容器管理的Bean,此時就需要直接獲取到對應的Bean。

本文為大家整理匯總了常見的獲取Bean的方式,并提供一些優劣分析,方便大家在使用到時有更好的選擇。同時,也會為大家適當的普及和拓展一些相關知識。

Spring的IoC容器

在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與ApplicationContext的區別

BeanFactory是Spring框架的基礎設施,面向Spring本身。ApplicationContext則面向使用Spring框架的開發者,幾乎所有的應用場合都可以直接使用ApplicationContext,而非底層的BeanFactory。

另外,ApplicationContext的初始化和BeanFactory有一個重大的區別:

BeanFactory在初始化容器時,并未實例化Bean,直到第一次訪問某個Bean時才實例目標Bean。這樣,我們就不能發現一些存在的Spring的配置問題。如果Bean的某一個屬性沒有注入,BeanFacotry加載后,直至第一次使用調用getBean方法才會拋出異常。

而ApplicationContext則在初始化應用上下文時就實例化所有單實例的Bean,相對應的,ApplicationContext的初始化時間會比BeanFactory長一些。

了解了上述的基本理論知識之后,我們就可以嘗試從IoC容器當中獲取Bean對象了。

方式一:通過BeanFactory獲取

通過BeanFactory來獲取Bean。基于xml配置文件的時代,可以通過如下方式獲得BeanFactory,再通過BeanFactory來獲得對應的Bean。

BeanFactory beanFactory = new XmlBeanFactory(new ClassPathResource("applicationContext.xml"));
UserInfo userInfo = (UserInfo) beanFactory.getBean("userInfo");

有一定編程年齡的程序員,應該對此還有一些印象。這種寫法估計也只會出現在古老的項目當中。鑒于xml形式配置文件已經被基于注解形式所替代,同時XmlBeanFactory也被標注為廢棄。此種方式不推薦使用。

其實,不推薦的理由還有一個,在上面已經提到,盡量不要使用BeanFactory,而應該使用ApplicationContext。

方式二:通過BeanFactoryAware獲取

在上面的方式中,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

在項目啟動時先獲取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對象。

方式四:通過繼承ApplicationObjectSupport

此種方式依舊是先獲得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

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方法來獲取。

方式六:通過WebApplicationContextUtils

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

通過實現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

使用ContextLoader提供的getCurrentWebApplicationContext方法,也是常用的獲取WebApplicationContext的一種方法。

具體實現代碼如下:

WebApplicationContext wac = ContextLoader.getCurrentWebApplicationContext();
wac.getBean(beanID);

該方法常見于SpringMVC實現的Web項目中。該方式是一種不依賴于Servlet,不需要注入的方式。但是需要注意一點,在服務器啟動時和Spring容器初始化時,不能通過該方法獲取Spring容器。

方式九:通過BeanFactoryPostProcessor

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容器。

責任編輯:武曉燕 來源: 程序新視界
相關推薦

2009-07-20 15:08:41

Spring實例化Be

2024-06-19 19:17:04

2022-06-23 10:47:57

Spring容器工具

2024-01-23 08:47:13

BeanSpring加載方式

2016-10-19 14:37:09

2022-11-17 15:17:12

Java數據結構Map

2025-04-25 11:25:00

SpringBean初始化

2023-02-24 11:49:02

ChatGPT首席信息官

2021-08-09 13:24:32

數據分析大數據小企業

2025-03-26 00:35:25

2022-06-28 10:22:00

機器學習網絡攻擊黑客

2023-11-03 08:19:18

SpringBean容器

2009-06-19 18:26:38

Spring事務配置

2011-02-28 13:51:30

Spring事物配置

2012-07-17 09:16:16

SpringSSH

2009-06-17 17:20:14

BeanFactorySpring

2011-11-25 10:25:27

SpringJava

2011-04-02 15:25:41

Spring

2022-03-16 11:11:37

SpringBean項目

2020-02-10 15:50:18

Spring循環依賴Java
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 日韩欧美成人精品 | 亚洲人a| 亚洲精品一区二区在线观看 | 丁香久久 | 亚洲国产成人精品女人久久久 | 成年视频在线观看福利资源 | 毛片免费观看 | 91人人看| 欧美一区二区三区在线视频 | 国产韩国精品一区二区三区 | 欧美激情一区二区 | 成人国产一区二区三区精品麻豆 | 久久久日韩精品一区二区三区 | aaa在线 | 国产黄色网 | 欧美日韩大陆 | 亚洲综合五月天婷婷 | 国产成人a亚洲精品 | av手机在线播放 | 综合久久av | 亚洲xxxxx| 中文字幕日韩欧美一区二区三区 | 日韩精品视频在线 | 国产亚洲一区二区三区在线观看 | 性网站免费 | 精品国产高清一区二区三区 | 免费h在线 | 午夜性色a√在线视频观看9 | 亚洲成人av | 国产专区在线 | 另类 综合 日韩 欧美 亚洲 | 亚洲网站在线播放 | 99久久成人 | 欧美日韩一二三区 | 国产精品视频一区二区三区四区国 | 欧美久久不卡 | 中文字幕在线三区 | 久久久久久久久久久久久久久久久久久久 | 韩国成人在线视频 | 亚洲国产情侣 | 国产精品免费视频一区 |