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

如何讓 Bean 深度感知 Spring 容器

開發(fā) 前端
雖然 Spring 中的 Bean 可以不用去感知 Spring 容器的存在,但是在實際開發(fā)中,我們往往還是需要 Spring 容器提供的各種能力,這樣就迫使我們的 Bean 不得不去感知到 Spring 容器的存在。

Spring 有一個特點,就是創(chuàng)建出來的 Bean 對容器是無感的,一個 Bean 是怎么樣被容器從一個 Class 整成一個 Bean 的,對于 Bean 本身來說是不知道的,當(dāng)然也不需要知道,也就是 Bean 對容器的存在是無感的。

但是有時候我們可能會遇到一些場景,這些場景讓我們?nèi)ジ兄萜鞯拇嬖冢筛缗e幾個例子:

  1. Spring 容器提供的功能不止 IoC、AOP 這些,常見的 I18N 也是 Spring 的能力之一,如果我們想要在自己的 Bean 中去使用 I18N,那就得去找 Spring,這樣就感知到了 Spring 容器的存在了。
  2. Spring 提供了資源加載器,如果我們想要使用這個資源加載器去加載配置,那就得去找 Spring 要,這樣就感知到了 Spring 容器的存在了。
  3. 想根據(jù) beanName 去 Spring 容器中查找 Bean,那不用多說,肯定得知道 Spring 容器的存在。
  4. ...

也就是說,雖然 Spring 中的 Bean 可以不用去感知 Spring 容器的存在,但是在實際開發(fā)中,我們往往還是需要 Spring 容器提供的各種能力,這樣就迫使我們的 Bean 不得不去感知到 Spring 容器的存在。

那么 Spring 中的 Bean 如何感知到 Spring 容器的存在呢?

1. Aware

Aware 本身就有感知的意思。

Spring Aware 是 Spring 框架中的一個特性,它允許我們的應(yīng)用程序或組件與 Spring 容器進(jìn)行交互。當(dāng)一個類實現(xiàn)了 Spring Aware 接口并注冊到 Spring 容器中時,該類就能夠感知到 Spring 容器的存在,并且可以獲取容器的一些資源或進(jìn)行一些特定的操作。

Spring Aware 接口包括了多個子接口,每個子接口對應(yīng)于不同的 Spring 容器資源或功能。

Aware 的實現(xiàn)有很多,大的方向來說主要有如下一些:

圖片圖片

每一個 Aware 的作用如下:

  • ApplicationEventPublisherAware:實現(xiàn)該接口的對象可以獲取事件發(fā)布的能力。
  • ServletContextAware:實現(xiàn)該接口的對象可以獲取到 ServletContext 對象。
  • MessageSourceAware:實現(xiàn)該接口的對象可以獲取到 MessageSource 對象,MessageSource 支持多消息源,主要用于主要用于國際化。
  • ResourceLoaderAware:實現(xiàn)該接口的對象可以獲取到一個 ResourceLoader,Spring ResourceLoader 則為我們提供了一個統(tǒng)一的 getResource() 方法來通過資源路徑檢索外部資源,例如文本文件、XML 文件、屬性文件或圖像文件等。
  • ApplicationStartupAware:實現(xiàn)該接口的對象可以獲取到一個 ApplicationStartup 對象,這個比較新,是 Spring 5.3 中新推出的,通過 ApplicationStartup 可以標(biāo)記應(yīng)用程序啟動期間的步驟,并收集有關(guān)執(zhí)行上下文或其處理時間的數(shù)據(jù)。
  • NotificationPublisherAware:實現(xiàn)該接的對象可以獲取到一個 NotificationPublisher 對象,通過該對象可以實現(xiàn)通知的發(fā)送。
  • EnvironmentAware:實現(xiàn)該接口的對象可以獲取到一個 Environment 對象,通過 Environment 可以獲取到容器的環(huán)境信息。
  • BeanFactoryAware:實現(xiàn)該接口的對象可以獲取到一個 BeanFactory 對象,通過 BeanFactory 可以完成 Bean 的查詢等操作。
  • ImportAware:實現(xiàn)該接口的對象可以獲取到一個 AnnotationMetadata 對象,ImportAware 接口是需要和 @Import 注解一起使用的。在 @Import 作為元注解使用時,通過 @Import 導(dǎo)入的配置類如果實現(xiàn)了 ImportAware 接口就可以獲取到導(dǎo)入該配置類接口的數(shù)據(jù)配置。
  • EmbeddedValueResolverAware:實現(xiàn)該接口的對象可以獲取到一個 StringValueResolver 對象,通過 StringValueResolver 對象,可以讀取到 Spring 容器中的 properties 配置的值(YAML 配置也可以)。
  • ServletConfigAware:實現(xiàn)該接口的對象可以獲取到一個 ServletConfig 對象,不過這個似乎沒什么用,我們很少自己去配置 ServletConfig。
  • LoadTimeWeaverAware:實現(xiàn)該接口的對象可以獲取到一個 LoadTimeWeaver 對象,通過該對象可以獲取加載 Spring Bean 時織入的第三方模塊,如 AspectJ 等。
  • BeanClassLoaderAware:實現(xiàn)該接口的對象可以獲取到一個 ClassLoader 對象,ClassLoader 能干嘛不需要我多說了吧。
  • BeanNameAware:實現(xiàn)該接口的對象可以獲取到一個當(dāng)前 Bean 的名稱。
  • ApplicationContextAware:實現(xiàn)該接口的對象可以獲取到一個 ApplicationContext 對象,通過 ApplicationContext 可以獲取容器中的 Bean、環(huán)境等信息。

通過實現(xiàn)這些接口,我們可以在應(yīng)用程序中獲取 Spring 容器提供的各種資源,并與容器進(jìn)行交互,以實現(xiàn)更靈活和可擴展的功能。

2. 實踐

舉兩個例子小伙伴們來感受下 Aware 的具體用法。

2.1 案例

例如我想在 Bean 中感知到當(dāng)前 Bean 的名字,那么我們可以按照如下方式來使用:

@Service
public class UserService implements BeanNameAware {
    private String beanName;
    @Override
    public void setBeanName(String name) {
        this.beanName = name;
    }

    @Override
    public String toString() {
        return "UserService{" +
                "beanName='" + beanName + '\'' +
                '}';
    }
}

讓當(dāng)前 bean 實現(xiàn) BeanNameAware 接口,并重寫 setBeanName 方法,這個方法會在 Spring 容器初始化 Bean 的時候自動被調(diào)用,我們就可以據(jù)此獲取到 bean 的名稱了。

再比如我想做一個工具 Bean,用來查找其他 Bean,那么我可以使用如下方式:

@Component
public class BeanUtils implements BeanFactoryAware {
    private static BeanFactory beanFactory;
    @Override
    public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
        this.beanFactory = beanFactory;
    }

    public static  <T> T getBean(Class<T> clazz) {
        return (T) beanFactory.getBean(clazz);
    }
}

讓當(dāng)前 Bean 實現(xiàn) BeanFactoryAware 接口并重寫 setBeanFactory 方法,在系統(tǒng)初始化當(dāng)前 Bean 的時候,會自動調(diào)用 setBeanFactory 方法,進(jìn)而將 beanFactory 變量傳進(jìn)來。

2.2 原理

當(dāng) Spring 容器創(chuàng)建一個 Bean 的時候,大致的流程是創(chuàng)建實例對象 -> 屬性填充 -> Bean 初始化

最后這個 Bean 的初始化,就是調(diào)用 init 方法、afterPropertiesSet 方法以及 BeanPostProcessor 中的方法的,如下:

protected Object initializeBean(String beanName, Object bean, @Nullable RootBeanDefinition mbd) {
 invokeAwareMethods(beanName, bean);
 Object wrappedBean = bean;
 if (mbd == null || !mbd.isSynthetic()) {
  wrappedBean = applyBeanPostProcessorsBeforeInitialization(wrappedBean, beanName);
 }
 try {
  invokeInitMethods(beanName, wrappedBean, mbd);
 }
 catch (Throwable ex) {
  throw new BeanCreationException(
    (mbd != null ? mbd.getResourceDescription() : null), beanName, ex.getMessage(), ex);
 }
 if (mbd == null || !mbd.isSynthetic()) {
  wrappedBean = applyBeanPostProcessorsAfterInitialization(wrappedBean, beanName);
 }
 return wrappedBean;
}

在這個方法一進(jìn)來,首先有一個 invokeAwareMethods,這個就是用來觸發(fā) Aware 的,來看下:

private void invokeAwareMethods(String beanName, Object bean) {
 if (bean instanceof Aware) {
  if (bean instanceof BeanNameAware beanNameAware) {
   beanNameAware.setBeanName(beanName);
  }
  if (bean instanceof BeanClassLoaderAware beanClassLoaderAware) {
   ClassLoader bcl = getBeanClassLoader();
   if (bcl != null) {
    beanClassLoaderAware.setBeanClassLoader(bcl);
   }
  }
  if (bean instanceof BeanFactoryAware beanFactoryAware) {
   beanFactoryAware.setBeanFactory(AbstractAutowireCapableBeanFactory.this);
  }
 }
}

小伙伴們可以看到,BeanNameAware、BeanClassLoaderAware 以及 BeanFactoryAware 這三種類型的 Aware 是在這里觸發(fā)的。

每種 Aware 因為功能不同,因此作用的時機也不同。

invokeAwareMethods 方法執(zhí)行完畢之后,接下來是執(zhí)行 applyBeanPostProcessorsBeforeInitialization 方法,這個我們之前分析過,這個方法最終會觸發(fā) BeanPostProcessor#postProcessBeforeInitialization 方法的執(zhí)行,而 BeanPostProcessor 有一個子類專門處理 Aware 的,就是 ApplicationContextAwareProcessor:

@Override
@Nullable
public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
 if (!(bean instanceof EnvironmentAware || bean instanceof EmbeddedValueResolverAware ||
   bean instanceof ResourceLoaderAware || bean instanceof ApplicationEventPublisherAware ||
   bean instanceof MessageSourceAware || bean instanceof ApplicationContextAware ||
   bean instanceof ApplicationStartupAware)) {
  return bean;
 }
 invokeAwareInterfaces(bean);
 return bean;
}
private void invokeAwareInterfaces(Object bean) {
 if (bean instanceof Aware) {
  if (bean instanceof EnvironmentAware environmentAware) {
   environmentAware.setEnvironment(this.applicationContext.getEnvironment());
  }
  if (bean instanceof EmbeddedValueResolverAware embeddedValueResolverAware) {
   embeddedValueResolverAware.setEmbeddedValueResolver(this.embeddedValueResolver);
  }
  if (bean instanceof ResourceLoaderAware resourceLoaderAware) {
   resourceLoaderAware.setResourceLoader(this.applicationContext);
  }
  if (bean instanceof ApplicationEventPublisherAware applicationEventPublisherAware) {
   applicationEventPublisherAware.setApplicationEventPublisher(this.applicationContext);
  }
  if (bean instanceof MessageSourceAware messageSourceAware) {
   messageSourceAware.setMessageSource(this.applicationContext);
  }
  if (bean instanceof ApplicationStartupAware applicationStartupAware) {
   applicationStartupAware.setApplicationStartup(this.applicationContext.getApplicationStartup());
  }
  if (bean instanceof ApplicationContextAware applicationContextAware) {
   applicationContextAware.setApplicationContext(this.applicationContext);
  }
 }
}

大家看下,這七種類型的 Aware 是在這里被觸發(fā)的。

另外像 ImportAware 是在 ImportAwareBeanPostProcessor#postProcessBeforeInitialization 方法中處理的;LoadTimeWeaverAware 是在 、LoadTimeWeaverAwareProcessor#postProcessBeforeInitialization 方法中處理的。

基本上,大部分的 Aware 接口都是在 BeanPostProcessor 中處理的。

責(zé)任編輯:武曉燕 來源: 江南一點雨
相關(guān)推薦

2022-06-23 10:47:57

Spring容器工具

2023-01-13 07:41:20

BeanSpring容器

2009-06-19 11:18:51

Factory BeaSpring配置

2022-12-27 08:12:27

IOC容器Bean

2021-03-08 08:40:25

Spring Bean 創(chuàng)建單例對象

2015-12-03 14:33:35

2017-04-14 08:58:55

深度學(xué)習(xí)感知機深度網(wǎng)絡(luò)

2020-10-14 10:25:20

深度學(xué)習(xí)機器學(xué)習(xí)神經(jīng)網(wǎng)絡(luò)

2024-05-29 08:19:03

2020-12-11 08:04:22

SpringAOPBean

2009-06-18 11:15:53

裝配beanxml配置Spring

2023-09-12 16:20:04

邊緣AI深度學(xué)習(xí)

2010-09-15 11:20:09

2024-05-28 07:55:31

SpringBean用域

2016-10-17 17:57:51

無線手勢感知無線WiFinger

2023-07-26 00:43:31

2020-06-16 13:22:22

AI創(chuàng)新深度學(xué)習(xí)

2022-05-27 08:25:55

容器Spring

2021-07-01 10:03:55

Distroless容器安全

2021-05-11 07:42:59

BeanSpring屬性
點贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 久久se精品一区精品二区 | 在线中文字幕第一页 | 美女人人操 | 91视视频在线观看入口直接观看 | 国产精品视频网站 | 中文字幕av网站 | 欧美精品久久久 | 亚洲免费在线观看 | 九九热在线视频免费观看 | av片免费 | 91久久精品国产91久久 | 欧美a在线看 | 中文字幕电影在线观看 | 欧美视频区 | 三级视频在线观看电影 | 欧美日韩一区在线观看 | 亚洲精品二区 | 亚洲网站在线播放 | 91免费入口 | 亚洲欧美国产一区二区三区 | 成人妇女免费播放久久久 | av看片| 午夜精品一区二区三区在线观看 | 欧美成人h版在线观看 | 99久久婷婷国产综合精品电影 | 欧美专区在线观看 | 成人在线观看欧美 | 91在线电影 | 色射综合| 日韩免费一区 | 91精品久久久久久久99 | 福利片在线| 国产在线a| 天天草av| 亚洲综合大片69999 | www.精品国产 | 午夜精品久久久久久不卡欧美一级 | 一区二区三区精品视频 | 日韩欧美三区 | 亚洲欧美视频一区 | 91精品国产乱码久久蜜臀 |