Spring扩展接口Aware家族

在Spring中 , 相信大家都见过以Aware结尾的接口 , aware:英文意思是有感知的 。 就是让bean在创建的过程中 , 能感知到容器的变化 。 最常见的就是ApplicationContextAware接口 , 这个接口一般在项目中怎么使用呢?接下来就来说一说 。
【Spring扩展接口Aware家族】大家在使用Spring开发的时候 , 可能会遇到这样一个问题 。 就是想要在普通类中获取spring的bean , 该怎么做呢?
我们常见的方法就是定义一个类 , 实现ApplicationContextAware接口 , 然后在类里面定义一个staticApplicationContextapplicationContext对象作为成员变量 , 在重写setApplicationContext(ApplicationContextapplicationContext)方法时 , 将这里的applicationContext赋值给类中的成员变量applicationContext对象 , 然后在定义另一个static方法 , 在其中通过applicationContext.getBean(StringbeanName)来获取bean对象 , 这样就可以在普通类中调用这个static方法来获取bean了 。 具体代码如下:
@ComponentpublicclassSpringContextHolderimplementsApplicationContextAware{privatestaticApplicationContextapplicationContext;/***实现ApplicationContextAware接口的context注入函数,将其存入静态变量*在项目启动的时候执行*@paramapplicationContext*@throwsBeansException*/@OverridepublicvoidsetApplicationContext(ApplicationContextapplicationContext)throwsBeansException{SpringContextHolder.applicationContext=applicationContext;}publicstaticApplicationContextgetApplicationContext(){returnapplicationContext;}/***根据name获取bean*@parambeanName*@param*@return*/publicstaticTgetBean(StringbeanName){return(T)getApplicationContext().getBean(beanName);}/***根据class获取Bean*@paramclazz*@param*@return*/publicstaticTgetBean(Classclazz){returngetApplicationContext().getBean(clazz);}/***根据name,以及Clazz返回指定的Bean*@parambeanName*@paramclazz*@param*@return*/publicstaticTgetBean(StringbeanName,Classclazz){returngetApplicationContext().getBean(beanName,clazz);}}源码分析
你可能很疑惑为什么这样就能在实现类中获取到Spring的容器ApplicationContext对象呢?这里我们来看下源码 , 入口同样也是refresh()方法 , 接着finishBeanFactoryInitialization(beanFactory);然后是getBean() , 继续调用doCreateBean() , 在AbstractAutowireCapableBeanFactory类的doCreateBean()方法里:
注:这些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容器内@ConfigurationpublicclassMyEnvironmentAwareimplementsEnvironmentAware{privateEnvironmentenvironment;@OverridepublicvoidsetEnvironment(Environmentenvironment){this.environment=environment;}@BeanpublicCustomServicecustomService(){StringuploadPath=environment.getProperty("file.uploadPath");System.out.println("uploadPath============"+uploadPath);//通过environment获取到系统属性Stringjava_home=environment.getProperty("JAVA_HOME");System.out.println("java_home================"+java_home);//获取到前缀是spring.redis.的属性列表值RelaxedPropertyResolverrelaxedPropertyResolver=newRelaxedPropertyResolver(environment,"spring.redis.");System.out.println("spring.redis.host=========="+relaxedPropertyResolver.getProperty("host"));System.out.println("spring.redis.timeout=========="+relaxedPropertyResolver.getProperty("timeout"));returnnewCustomService();}}/***注意该类必须在ioc容器中使用 , 否则EmbeddedValueResolverAware不会注入进来*EmbeddedValueResolverAware只能读取配置文件的属性值*/@ComponentpublicclassPropertiesUtilimplementsEmbeddedValueResolverAware{privateStringValueResolverresolver;@OverridepublicvoidsetEmbeddedValueResolver(StringValueResolverresolver){this.resolver=resolver;}publicStringgetPropertiesValue(Stringkey){//StringValueResolver还可以解析spel表达式returnresolver.resolveStringValue("${"+key+


    推荐阅读