spring environment_Spring扩展接口Aware家族

在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了。具体代码如下:

@Componentpublic 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      * @return     */    public static  T getBean(String beanName){        return (T) getApplicationContext().getBean(beanName);    }    /**     * 根据class获取Bean     * @param clazz     * @param      * @return     */    public static  T getBean(Class clazz){        return getApplicationContext().getBean(clazz);    }    /**     * 根据name,以及Clazz返回指定的Bean     * @param beanName     * @param clazz     * @param      * @return     */    public static  T getBean(String beanName,Class clazz){        return getApplicationContext().getBean(beanName,clazz);    }}

源码分析

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

30aab6b3145f421cbef79637e719463e

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

53a066ab7bfe4140bd74730209f03078

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

b47ea93824fc46c1842a3b78983da6f6

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

67495c69d3b940f2960b51b922acb4cb
ecf0b8e0405a40c3aa943c1efd93e762

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

e087ac5733cd4d32a8c7d44950c4cf73

对于其他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容器内@Configurationpublic 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只能读取配置文件的属性值 */@Componentpublic 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+"}");    }}
@Componentpublic 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);    }}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值