日韩成人免费在线_国产成人一二_精品国产免费人成电影在线观..._日本一区二区三区久久久久久久久不

當(dāng)前位置:首頁 > 科技  > 軟件

如何讓 Bean 深度感知 Spring 容器

來源: 責(zé)編: 時間:2023-11-03 09:16:52 279觀看
導(dǎo)讀Spring 有一個特點,就是創(chuàng)建出來的 Bean 對容器是無感的,一個 Bean 是怎么樣被容器從一個 Class 整成一個 Bean 的,對于 Bean 本身來說是不知道的,當(dāng)然也不需要知道,也就是 Bean 對容器的存在是無感的。但是有時候我們可能

Spring 有一個特點,就是創(chuàng)建出來的 Bean 對容器是無感的,一個 Bean 是怎么樣被容器從一個 Class 整成一個 Bean 的,對于 Bean 本身來說是不知道的,當(dāng)然也不需要知道,也就是 Bean 對容器的存在是無感的。xz228資訊網(wǎng)——每日最新資訊28at.com

但是有時候我們可能會遇到一些場景,這些場景讓我們?nèi)ジ兄萜鞯拇嬖冢筛缗e幾個例子:xz228資訊網(wǎng)——每日最新資訊28at.com

  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 容器的存在。xz228資訊網(wǎng)——每日最新資訊28at.com

那么 Spring 中的 Bean 如何感知到 Spring 容器的存在呢?xz228資訊網(wǎng)——每日最新資訊28at.com

1. Aware

Aware 本身就有感知的意思。xz228資訊網(wǎng)——每日最新資訊28at.com

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

Spring Aware 接口包括了多個子接口,每個子接口對應(yīng)于不同的 Spring 容器資源或功能。xz228資訊網(wǎng)——每日最新資訊28at.com

Aware 的實現(xiàn)有很多,大的方向來說主要有如下一些:xz228資訊網(wǎng)——每日最新資訊28at.com

圖片圖片xz228資訊網(wǎng)——每日最新資訊28at.com

每一個 Aware 的作用如下:xz228資訊網(wǎng)——每日最新資訊28at.com

  • 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 容器提供的各種資源,并與容器進行交互,以實現(xiàn)更靈活和可擴展的功能。xz228資訊網(wǎng)——每日最新資訊28at.com

2. 實踐

舉兩個例子小伙伴們來感受下 Aware 的具體用法。xz228資訊網(wǎng)——每日最新資訊28at.com

2.1 案例

例如我想在 Bean 中感知到當(dāng)前 Bean 的名字,那么我們可以按照如下方式來使用:xz228資訊網(wǎng)——每日最新資訊28at.com

@Servicepublic 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 的名稱了。xz228資訊網(wǎng)——每日最新資訊28at.com

再比如我想做一個工具 Bean,用來查找其他 Bean,那么我可以使用如下方式:xz228資訊網(wǎng)——每日最新資訊28at.com

@Componentpublic 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 方法,進而將 beanFactory 變量傳進來。xz228資訊網(wǎng)——每日最新資訊28at.com

2.2 原理

當(dāng) Spring 容器創(chuàng)建一個 Bean 的時候,大致的流程是創(chuàng)建實例對象 -> 屬性填充 -> Bean 初始化xz228資訊網(wǎng)——每日最新資訊28at.com

最后這個 Bean 的初始化,就是調(diào)用 init 方法、afterPropertiesSet 方法以及 BeanPostProcessor 中的方法的,如下:xz228資訊網(wǎng)——每日最新資訊28at.com

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;}

在這個方法一進來,首先有一個 invokeAwareMethods,這個就是用來觸發(fā) Aware 的,來看下:xz228資訊網(wǎng)——每日最新資訊28at.com

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ā)的。xz228資訊網(wǎng)——每日最新資訊28at.com

每種 Aware 因為功能不同,因此作用的時機也不同。xz228資訊網(wǎng)——每日最新資訊28at.com

invokeAwareMethods 方法執(zhí)行完畢之后,接下來是執(zhí)行 applyBeanPostProcessorsBeforeInitialization 方法,這個我們之前分析過,這個方法最終會觸發(fā) BeanPostProcessor#postProcessBeforeInitialization 方法的執(zhí)行,而 BeanPostProcessor 有一個子類專門處理 Aware 的,就是 ApplicationContextAwareProcessor:xz228資訊網(wǎng)——每日最新資訊28at.com

@Override@Nullablepublic 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ā)的。xz228資訊網(wǎng)——每日最新資訊28at.com

另外像 ImportAware 是在 ImportAwareBeanPostProcessor#postProcessBeforeInitialization 方法中處理的;LoadTimeWeaverAware 是在 、LoadTimeWeaverAwareProcessor#postProcessBeforeInitialization 方法中處理的。xz228資訊網(wǎng)——每日最新資訊28at.com

基本上,大部分的 Aware 接口都是在 BeanPostProcessor 中處理的。xz228資訊網(wǎng)——每日最新資訊28at.com

xz228資訊網(wǎng)——每日最新資訊28at.com

本文鏈接:http://www.www897cc.com/showinfo-26-16740-0.html如何讓 Bean 深度感知 Spring 容器

聲明:本網(wǎng)頁內(nèi)容旨在傳播知識,若有侵權(quán)等問題請及時與本網(wǎng)聯(lián)系,我們將在第一時間刪除處理。郵件:2376512515@qq.com

上一篇: 用 Java 深入研究樹,你了解多少?

下一篇: 如何在C#客戶端程序中無縫集成Python算法

標(biāo)簽:
  • 熱門焦點
Top 主站蜘蛛池模板: 隆化县| 龙里县| 施甸县| 周宁县| 西乌| 赤峰市| 二连浩特市| 迭部县| 昭平县| 建瓯市| 繁昌县| 嘉义市| 海晏县| 鄂托克前旗| 上林县| 沧源| 大田县| 石屏县| 满城县| 金阳县| 景宁| 绥滨县| 镇江市| 邓州市| 井研县| 洛隆县| 娱乐| 灌云县| 林西县| 鄯善县| 民和| 茶陵县| 冷水江市| 金门县| 顺义区| 射洪县| 宜黄县| 营口市| 乐昌市| 新闻| 湖北省|