SpringBoot ApplicationContextAware拓展接口使用详解

作者:ForestSpringH 时间:2022-04-01 23:20:24 

近日沉醉于熟悉公司新项目的过程,不断地接触新的应用场景与实现技术,对于我是一种学不来的进步,实践是检验真理的唯一标准。我们今天就浅浅的谈一谈springboot提供的16个拓展接口之一的ApplicationContextAware接口的应用场景与实际作用!

ApplicationContextAware接口:

public interface ApplicationContextAware extends Aware {
   void setApplicationContext(ApplicationContext applicationContext) throws BeansException;
}

首先Aware接口就知道这是springboot扩展给用户使用的,这里提供了方法setApplicationContext,参数就是传递spring容器上下文对象进来,我们可以接收这个上下文对象,我们要想知道获取spring容器上下文ApplicationContext具体有什么作用,这才是扩展接口的目的所在,获取上下文根据上下文的特性做一些事情。

我们来看ApplicationContext对象的方法:

SpringBoot ApplicationContextAware拓展接口使用详解

来看看AbstractApplicationContext实现类的方法:

public Object getBean(String name) throws BeansException {this.assertBeanFactoryActive();return this.getBeanFactory().getBean(name);}
   public <T> T getBean(String name, Class<T> requiredType) throws BeansException {this.assertBeanFactoryActive();return this.getBeanFactory().getBean(name, requiredType);}
   public Object getBean(String name, Object... args) throws BeansException {this.assertBeanFactoryActive();return this.getBeanFactory().getBean(name, args);}
   public <T> T getBean(Class<T> requiredType) throws BeansException {this.assertBeanFactoryActive();return this.getBeanFactory().getBean(requiredType);}
   public <T> T getBean(Class<T> requiredType, Object... args) throws BeansException {this.assertBeanFactoryActive();return this.getBeanFactory().getBean(requiredType, args);}
   public <T> ObjectProvider<T> getBeanProvider(Class<T> requiredType) {this.assertBeanFactoryActive();return this.getBeanFactory().getBeanProvider(requiredType);}
   public <T> ObjectProvider<T> getBeanProvider(ResolvableType requiredType) {this.assertBeanFactoryActive();return this.getBeanFactory().getBeanProvider(requiredType);}
   public boolean containsBean(String name) {return this.getBeanFactory().containsBean(name);}
   public boolean isSingleton(String name) throws NoSuchBeanDefinitionException {this.assertBeanFactoryActive();return this.getBeanFactory().isSingleton(name);}
   public boolean isPrototype(String name) throws NoSuchBeanDefinitionException {this.assertBeanFactoryActive();return this.getBeanFactory().isPrototype(name);}

这里我们可以发现 getBean()方法很眼熟,因为在最最开始学习spring时没有用spring的脚手架创建项目,我们获取bean的方法通常是classPathContextLoader扫描bean的xml文件解析组成ApplicationCOntext对象,再调用它的getBean方法获取实例bean。

由此可以发现我们主要的应用途径就是使用这个getBean的方法,那么动态的注入bean我们通过很多方法就能实现,所以这里不难想到,静态方法中无法使用注入的bean的问题。

其次我们来复现这个问题,大家来看如下的代码:

public class JsonGetter {
@Resource
private UuidGetter uuidGetter;
public static string Test(){
      return uuidGetter.getUuid();
}
public static JsONobject set0bjectToJsonObject(object data){
      return JsoNobject.parseObject(String.valueof(JsONObject.toJSON(data)));
}
public static JsONObject setStringTO3son0bject(String data) { return JsONObject.parseObject(data);
}

我们发现在静态的Test方法中调用注入的bean直接报错,这里解释一下:归功于类的加载机制与加载顺序,静态属性与静态代码块最先加载(static静态优先),这里加载静态方法是没有bean实例给你用的,自然会报错。

如何解决?我们可以采取Spring获取bean对象时调用getBean方法的思路,在容器加载时将spring容器的上下文进行静态存储:

@Component
@Lazy(value = false)
public class SpringContextHolder implements ApplicationContextAware, DisposableBean {
   /**
    * 将上下文静态设置,在初始化组件时就进行静态上下文的覆盖(这个覆盖是将远spring容器的上下文对象引用加到我们预定设置)
    */
   private static ApplicationContext applicationContext = null;
   public static ApplicationContext getApplicationContext() {
       assertContextInjected();
       return applicationContext;
   }
   @SuppressWarnings("unchecked")
   public static <T> T getBean(String name) {
       assertContextInjected();
       return (T) applicationContext.getBean(name);
   }
   public static  <T> T getBean(Class<T> beanType) {
       assertContextInjected();
       return applicationContext.getBean(beanType);
   }
   @Override
   public void setApplicationContext(@NotNull ApplicationContext applicationContext) throws BeansException {
       SpringContextHolder.applicationContext = applicationContext;
   }
   @Override
   public void destroy() {
       applicationContext = null;
   }
   private static void assertContextInjected() {
       Assert.notNull(applicationContext,
               "applicationContext属性未注入, 请在applicationContext.xml中定义SpringContextHolder.");
   }
   public static void pushEvent(ApplicationEvent event){
       assertContextInjected();
       applicationContext.publishEvent(event);
   }
}

这里只需要关注的是静态成员变量ApplicationContext的定义、赋值与验证:

/**
    * 将上下文静态设置,在初始化组件时就进行静态上下文的覆盖(这个覆盖是将远spring容器的上下文对象引用加到我们预定设置)
    */
   private static ApplicationContext applicationContext = null;

重写扩展接口的方法,实现静态上下文的覆盖:

@Override
   public void setApplicationContext(@NotNull ApplicationContext applicationContext) throws BeansException {
       SpringContextHolder.applicationContext = applicationContext;
   }

将获取它的方法公有修饰,便于共享:

public static ApplicationContext getApplicationContext() {
       assertContextInjected();
       return applicationContext;
   }

写到这里还是不明白,这么定义一个组件,将spring上下文对象静态覆盖到底有何作用?

不要慌,我们来看看这个类的这个方法:

public class AppContext {
   static transient ThreadLocal<Map<String, String>> contextMap = new ThreadLocal<>();
   ......省略n行业务代码
   public static void fillLoginContext() {
       DingAppInfo appInfo = SpringContextHolder.getBean(DingAppInfoService.class).findAppInfo(APP_CODE);
       setDingVerifyInfo(appInfo);
       CloudChatAppInfo cloudChatAppInfo = SpringContextHolder.getBean(CloudChatAppInfoService.class).findAppInfo(APP_CODE);
       setCloudChatInfo(cloudChatAppInfo);
   }
   public static void clear() {
       contextMap.remove(); //本地线程的remove方法极其重要,注意每次给它使用之后一定要调用remove清理,防止内存泄露。
   }
}

我们发现上例代码中进行了查库的操作:

DingAppInfo appInfo = SpringContextHolder.getBean(DingAppInfoService.class).findAppInfo(APP_CODE);

这里是不是就解决了最初我们的问题?

实现了扩展applicationContextAware接口,解决静态方法无法获取bean实例调用其方法的问题?

来源:https://blog.csdn.net/m0_59588838/article/details/129890653

标签:SpringBoot,ApplicationContextAware,拓展接口
0
投稿

猜你喜欢

  • C#编程调用Cards.dll实现图形化发牌功能示例

    2022-10-24 12:02:54
  • Android集成微信登录的步骤详解

    2023-01-12 15:12:45
  • 深入学习java8 中的CompletableFuture

    2022-05-19 04:44:26
  • spring-boot读取props和yml配置文件的方法

    2022-06-01 14:54:26
  • Java多线程Thread基础学习

    2023-04-17 17:12:21
  • EditPlus运行java时从键盘输入数据的操作方法

    2023-04-16 21:18:45
  • c#循环中产生伪随机数

    2023-05-17 23:09:40
  • C语言malloc分配问题详解

    2023-07-22 05:10:34
  • 浅谈Java三大框架与应用

    2023-04-16 18:25:01
  • C# 泛型List排序的实现

    2021-09-22 07:03:16
  • 基于Avalonia实现自定义弹窗的示例详解

    2022-02-27 16:30:45
  • C#实现的MD5加密功能与用法示例

    2023-06-11 09:08:21
  • 深入探究Java线程的创建与构造方法

    2023-05-29 22:02:45
  • C# WinForm调用Shell_NotifyIcon的示例代码

    2021-07-17 16:04:41
  • C#导出生成excel文件的方法小结(xml,html方式)

    2023-10-03 16:32:26
  • C#使用DropDownList绑定添加新数据的方法汇总

    2023-05-08 03:20:56
  • Java客户端利用Jedis操作redis缓存示例代码

    2021-05-31 03:43:42
  • C#快速实现IList非泛型类接口的自定义类作为数据源

    2022-10-09 20:28:24
  • C# 获取 PC 序列号的方法示例

    2022-11-19 01:40:57
  • C#获取鼠标在listview右键点击单元格的内容方法

    2023-10-26 12:40:47
  • asp之家 软件编程 m.aspxhome.com