快捷搜索:  汽车  科技

springboot 接口顺序(Spring扩展接口Aware家族)

springboot 接口顺序(Spring扩展接口Aware家族)在其中又调用了多个Aware实现类的set()方法,其中就包括我们前面讲的ApplicationContextAware的实现类,就是在这里将容器applicationContext对象设置到我们自定义的类中的,这样就可以让我们的bean在生成的时候能感知到容器的变化。接着又调用了ApplicationContextAwareProcessor类中的invokeAwareInterfaces()方法:你可能很疑惑为什么这样就能在实现类中获取到Spring的容器ApplicationContext对象呢?这里我们来看下源码,入口同样也是refresh()方法,接着finishBeanFactoryInitialization(beanFactory);然后是getBean(),继续调用doCreateBean(),在AbstractAutowireCapableBeanFactory类的do

在Spring中,相信大家都见过以Aware结尾的接口,aware:英文意思是有感知的。就是让bean在创建的过程中,能感知到容器的变化。最常见的就是ApplicationContextAware接口,这个接口一般在项目中怎么使用呢?接下来就来说一说。

大家在使用Spring开发的时候,可能会遇到这样一个问题。就是想要在普通类中获取spring的bean,该怎么做呢?

我们常见的方法就是定义一个类,实现applicationContextAware接口,然后在类里面定义一个static ApplicationContext applicationContext 对象作为成员变量,在重写setApplicationContext(ApplicationContext applicationContext)方法时,将这里的applicationContext赋值给类中的成员变量applicationContext对象,然后在定义另一个static方法,在其中通过applicationContext.getBean(String beanName)来获取bean对象,这样就可以在普通类中调用这个static方法来获取bean了。具体代码如下:

@Component public class SpringContextHolder implements ApplicationContextAware { private static ApplicationContext applicationContext; /** * 实现ApplicationContextAware接口的context注入函数 将其存入静态变量 * 在项目启动的时候执行 * @param applicationContext * @throws BeansException */ @Override public void setApplicationContext(ApplicationContext applicationContext) throws BeansException { SpringContextHolder.applicationContext = applicationContext; } public static ApplicationContext getApplicationContext(){ return applicationContext; } /** * 根据name获取bean * @param beanName * @param <T> * @return */ public static <T> T getBean(String beanName){ return (T) getApplicationContext().getBean(beanName); } /** * 根据class获取Bean * @param clazz * @param <T> * @return */ public static <T> T getBean(Class<T> clazz){ return getApplicationContext().getBean(clazz); } /** * 根据name 以及Clazz返回指定的Bean * @param beanName * @param clazz * @param <T> * @return */ public static <T> T getBean(String beanName Class<T> clazz){ return getApplicationContext().getBean(beanName clazz); } }

源码分析

你可能很疑惑为什么这样就能在实现类中获取到Spring的容器ApplicationContext对象呢?这里我们来看下源码,入口同样也是refresh()方法,接着finishBeanFactoryInitialization(beanFactory);然后是getBean(),继续调用doCreateBean(),在AbstractAutowireCapableBeanFactory类的doCreateBean()方法里:

springboot 接口顺序(Spring扩展接口Aware家族)(1)

调用了initializeBean()方法,我们可以看到在initializeBean()里先后调用了invokeAwareMethods(beanName bean);和applyBeanPostProcessorsBeforeInitialization(wrappedBean beanName);方法

springboot 接口顺序(Spring扩展接口Aware家族)(2)

在invokeAwareMethods调用了多个Aware实现类的set()方法:

springboot 接口顺序(Spring扩展接口Aware家族)(3)

接着又调用了ApplicationContextAwareProcessor类中的invokeAwareInterfaces()方法:

springboot 接口顺序(Spring扩展接口Aware家族)(4)

springboot 接口顺序(Spring扩展接口Aware家族)(5)

在其中又调用了多个Aware实现类的set()方法,其中就包括我们前面讲的ApplicationContextAware的实现类,就是在这里将容器applicationContext对象设置到我们自定义的类中的,这样就可以让我们的bean在生成的时候能感知到容器的变化。

springboot 接口顺序(Spring扩展接口Aware家族)(6)

对于其他Aware接口的实现类也是如此,都是在这里进行赋值传递。

注:这些Aware接口的操作是发生在bean初始化之前的。即在bean调用afterPropertiesSet()方法和init-method方法之前。

Aware接口家族

BeanNameAware:设置bean的名称

BeanClassLoaderAware:设置bean的类加载器

BeanFactoryAware:设置beanFactory容器

EnvironmentAware:设置Environment对象,可以用来获取环境变量和配置文件属性

EmbeddedValueResolverAware:设置StringValueResolver对象,可以用来获取配置文件属性

ResourceLoaderAware:设置ResourceLoader对象,可以用来读取各种不同形式的资源文件,如classpath、jar、file等等

ApplicationEventPublisherAware:设置ApplicationEventPublisher对象,可以发送事件通知

MessageSourceAware:设置MessageSource对象,可以用来进行国际化的实现

ApplicationContextAware:设置ApplicationContext容器,可以用来获取bean对象

针对以上的几个Aware接口,我们看一下几个示例:

/** * 凡注册到Spring容器内的bean,实现了EnvironmentAware接口重写setEnvironment方法后, * 在工程启动时可以获得application.properties的配置文件配置的属性值。 * Note:@Controller @Service 等被Spring管理的类都支持 */ //@Configuration注解在SpringBoot里面相当于Spring的XML文件里的beans标签一样, // 而@Bean注解相当于XML文件里的bean标签,代表该类会被加载到Spring的IOC容器内 @Configuration public class MyEnvironmentAware implements EnvironmentAware { private Environment environment; @Override public void setEnvironment(Environment environment) { this.environment = environment; } @Bean public CustomService customService(){ String uploadPath = environment.getProperty("file.uploadPath"); System.out.println("uploadPath============" uploadPath); //通过 environment 获取到系统属性 String java_home = environment.getProperty("JAVA_HOME"); System.out.println("java_home================" java_home); //获取到前缀是spring.redis.的属性列表值 RelaxedPropertyResolver relaxedPropertyResolver = new RelaxedPropertyResolver(environment "spring.redis."); System.out.println("spring.redis.host==========" relaxedPropertyResolver.getProperty("host")); System.out.println("spring.redis.timeout==========" relaxedPropertyResolver.getProperty("timeout")); return new CustomService(); } }

/** * 注意该类必须在ioc容器中使用,否则EmbeddedValueResolverAware不会注入进来 * EmbeddedValueResolverAware只能读取配置文件的属性值 */ @Component public class PropertiesUtil implements EmbeddedValueResolverAware { private StringValueResolver resolver; @Override public void setEmbeddedValueResolver(StringValueResolver resolver) { this.resolver = resolver; } public String getPropertiesValue(String key) { //StringValueResolver还可以解析spel表达式 return resolver.resolveStringValue("${" key "}"); } }

@Component public class MyMessageSourceAware implements MessageSourceAware { private MessageSource messageSource; @Override public void setMessageSource(MessageSource messageSource) { this.messageSource = messageSource; } public void printMessage(){ String name = messageSource.getMessage("user.name" null Locale.US); System.out.println("user name (English) : " name); String namechinese = messageSource.getMessage("user.name" null Locale.SIMPLIFIED_CHINESE); System.out.println("user name (Chinese) : " namechinese); } }

猜您喜欢: