Spring中@RefreshScope注解的处理方法详解

 更新时间:2023年10月25日 08:57:04   作者:葡萄晓虎  
这篇文章主要介绍了Spring中@RefreshScope注解的处理方法详解,spring启动时会调用ClassPathBeanDefinitionScanner.java类中的doScan()对包路径下的所有class进行扫描,获取bean的定义,同时对bean的@RefreshScope(@Scope的父类)进行处理,需要的朋友可以参考下

@RefreshScope注解

spring启动时会调用ClassPathBeanDefinitionScanner.java类中的doScan()对包路径下的所有class进行扫描,获取bean的定义,同时对bean的@RefreshScope(@Scope的父类)进行处理。

  1. 获取作用在类上的@RefreshScope(@Scope的父类)注解元数据,封装成ScopeMetadata对象,同时声明bean的作用域candidate.setScope(scopeMetadata.getScopeName());
  2. 根据scopeMetadata即@Scope对bean进行代理,使用ScopedProxyFactoryBean类创建RootBeanDefinition代替原理的bean定义。将被代理的原始bean定义注册到容器中(scopedTarget.configController -> ConfigController),再将代理的bean定义注册到容器中(configController -> ScopedProxyFactoryBean)

处理这个使用@RefreshScope注解的bean的是RefreshScope组件,可以看到父类GenericScope实现了BeanDefinitionRegistryPostProcessor和BeanFactoryPostProcessor接口,这是容器的后置处理器,当容器启动时会先调用BeanDefinitionRegistryPostProcessor接口中的postProcessBeanDefinitionRegistry()方法,再BeanFactoryPostProcessor接口中的postProcessBeanFactory()方法。

public class RefreshScope extends GenericScope
		implements ApplicationContextAware, Ordered{...}
public class GenericScope implements Scope, BeanFactoryPostProcessor,
		BeanDefinitionRegistryPostProcessor, DisposableBean{...}        

获取容器中包含的所有的bean定义,再通过bean装饰的bean定义获取作用域,判断是否为RefreshScope能处理的作用域,在构造RefreshScope实例时设置了name为"refresh"getName().equals(root.getDecoratedDefinition().getBeanDefinition().getScope())。ps:这是个细节,在被装饰的bean定义的scope是"refresh",而装饰者bean的scope是"";之后处理的时候先处理装饰者bean,在处理原始bean,就是通过这个scope来判断的。如果是的话将bean的class设置为LockedScopedProxyFactoryBean,将当前RefreshScope实例对象设置为构造函数的参数。看下LockedScopedProxyFactoryBean这个类:

public static class LockedScopedProxyFactoryBean<S extends GenericScope>
			extends ScopedProxyFactoryBean implements MethodInterceptor{...}
public class ScopedProxyFactoryBean extends ProxyConfig implements FactoryBean<Object>, BeanFactoryAware{...}            

可以看到LockedScopedProxyFactoryBean的父类实现了BeanFactoryAware接口,因此在bean实例化的时候会调用实现的setBeanFactory()方法,这时会通过代理工厂创建代理对象,即生成LockedScopedProxyFactoryBean类的代理对象。由于LockedScopedProxyFactoryBean类实现了Advised接口,因此可以作为代理对象的拦截器,当调用代理对象的方法时拦截器就会起作用。

public void setBeanFactory(BeanFactory beanFactory) {
......
    ProxyFactory pf = new ProxyFactory();
    pf.copyFrom(this);
    pf.setTargetSource(this.scopedTargetSource);
.....
    this.proxy = pf.getProxy(cbf.getBeanClassLoader());
}
public void setBeanFactory(BeanFactory beanFactory) {
    super.setBeanFactory(beanFactory);
    Object proxy = getObject();
    if (proxy instanceof Advised) {
        Advised advised = (Advised) proxy;
        advised.addAdvice(0, this);
    }
}

总结下@RefreshScope注解的bean实例化的过程:

  1. 使用了@RefreshScope注解的bean的定义会被设置成LockedScopedProxyFactoryBean.class
  2. 创建bean时先生成LockedScopedProxyFactoryBean的实例对象
  3. 由于LockedScopedProxyFactoryBean实现了BeanFactoryAware接口,会调用setBeanFactory()
  4. 在setBeanFactory()方法中会通过cglib创建LockedScopedProxyFactoryBean对象的代理对象,FactoryBean接口的getObject()会返回此代理对象,因此最终得到的就是LockedScopedProxyFactoryBean对象的代理对
  5. 由于代理对象为LockedScopedProxyFactoryBean的子类,因此实现了Advised接口,可以将调用setBeanFactory()方法的LockedScopedProxyFactoryBean对象作为拦截器保存下来

当容器启动完成刷新后会发布ContextRefreshedEvent事件,RefreshScope类中的start()使用了@EventListener注解,可以作为监听器对ContextRefreshedEvent事进行处理。找到scope是"refresh"的bean的定义,再走Scope的代码块进行实例化。

protected <T> T doGetBean(final String name, @Nullable final Class<T> requiredType,
			@Nullable final Object[] args, boolean typeCheckOnly) throws BeansException {
......
if (mbd.isSingleton()) {
    ......
}
else if (mbd.isPrototype()) {
    ......
}
else {
    String scopeName = mbd.getScope();
    //根据name获取Scope,当前一共有四个
    //refresh -> {RefreshScope@5469}
    //request -> {RequestScope@7462} 
    //session -> {SessionScope@7464} 
    //application -> {ServletContextScope@7466} 
    final Scope scope = this.scopes.get(scopeName);
    if (scope == null) {
        throw new IllegalStateException("No Scope registered for scope name '" + scopeName + "'");
    }
    try {
        Object scopedInstance = scope.get(beanName, () -> {
            beforePrototypeCreation(beanName);
            try {
                return createBean(beanName, mbd, args);
            }
            finally {
                afterPrototypeCreation(beanName);
            }
        });
        bean = getObjectForBeanInstance(scopedInstance, name, beanName, mbd);
    }
}
......
}

获取到RefreshScope对象后就会调用其中的get()方法进行初始化。先创建BeanLifecycleWrapper对象,包含了bean的name和创建bean的工厂方法(lamdba表达式),再调用BeanLifecycleWrapper类中的getBean()方法获取bean的实例对象,此时调用this.bean = this.objectFactory.getObject();工厂方法来创建bean,则回到createBean(beanName, mbd, args);方法完成bean的创建。接下来看看方法是怎么调用的。

  1. 获取被代理原始bean的实例对象
  2. 创建拦截器链,找到之前加入的LockedScopedProxyFactoryBean拦截器,调用invoke()方法进行处理
  3. 获取代理对象,即LockedScopedProxyFactoryBean的代理对象;获取读写锁的读锁,当前配置刷新时,则无法获取
  4. 以反射方式调用目标方法method.invoke(target, args)

再看看配置怎么刷新到bean当中的。当配置源变更时方法调用链如下:

  • NacosContextRefresher#registerNacosListener()注册监听器(receiveConfigInfo())
  • ApplicationEventPublisher.java#publishEvent()发布RefreshEvent事件
  • SimpleApplicationEventMulticaster#multicastEvent()方法广播事件
  • RefreshEventListener#handle(RefreshEvent event)处理RefreshEvent事件
  • ContextRefresher#refresh()进行处理,做了三件事
    • 刷新配置源
    • 发布EnvironmentChangeEvent事件,重新生成配置源Bean(@ConfigurationProperties注解Bean)
    • 将配置写到@RefreshScope注解标注的Bean

开始分析怎么刷新到@RefreshScope注解标注的Bean:将保存原始被代理的Bean的BeanLifecycleWrapperCache中清空,获取写锁(此时其他线程无法获取读锁),调用每个BeanLifecycleWrapper的destroy()方法进行销毁。缓存被清空了,如果重新获取Bean就得重新创建,此时就会进行Bean的属性填充,就可以从属性源中获取新的属性了

到此这篇关于Spring中@RefreshScope注解的处理方法详解的文章就介绍到这了,更多相关Spring的@RefreshScope注解内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java8使用Supplier启动ScheduledThread代码实例

    Java8使用Supplier启动ScheduledThread代码实例

    这篇文章主要介绍了Java8使用Supplier启动ScheduledThread详解,项目开启立即启动定时任务是很多项目都会遇到的一个需求,如何利用Java提供的函数优雅的写出来十分考验一个人的功底,需要的朋友可以参考下
    2024-01-01
  • Java排序算法之堆排思想及代码实现

    Java排序算法之堆排思想及代码实现

    今天小编就为大家分享一篇关于Java排序算法之堆排思想及代码实现,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2019-01-01
  • mybatis动态拼接实现有条件的插入

    mybatis动态拼接实现有条件的插入

    这篇文章主要介绍了mybatis动态拼接实现有条件的插入,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-02-02
  • Java中的日期和时间类以及Calendar类用法详解

    Java中的日期和时间类以及Calendar类用法详解

    这篇文章主要介绍了Java中的日期和时间类以及Calendar类用法详解,是Java入门学习中的基础知识,需要的朋友可以参考下
    2015-09-09
  • 详解MyBatis中Executor执行SQL语句的过程

    详解MyBatis中Executor执行SQL语句的过程

    MyBatis中获取SqlSession时会创建执行器Executor并存放在SqlSession中,本篇文章将以MapperMethod的execute() 方法作为起点,对MyBatis中的一次实际执行请求进行说明,并结合源码对执行器Executor的原理进行阐释
    2023-07-07
  • java中PreparedStatement和Statement详细讲解

    java中PreparedStatement和Statement详细讲解

    这篇文章主要介绍了java中PreparedStatement和Statement详细讲解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-11-11
  • Java数据结构之选择排序算法的实现与优化

    Java数据结构之选择排序算法的实现与优化

    选择排序:(Selection sort)是一种简单直观的排序算法,也是一种不稳定的排序方法。本文主要为大家介绍一下选择排序的实现与优化,希望对大家有所帮助
    2023-01-01
  • 详解Java内存泄露的示例代码

    详解Java内存泄露的示例代码

    这篇文章通过一个Demo来简要介绍下ThreadLocal和ClassLoader导致内存泄露最终OutOfMemory的场景。下面通过示例代码给大家分享Java内存泄露的相关知识,感兴趣的朋友一起看看吧
    2017-12-12
  • java循环结构、数组的使用小结

    java循环结构、数组的使用小结

    这篇文章主要介绍了java循环结构、数组的使用小结,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-09-09
  • jd-easyflow中inclusive的用法示例小结

    jd-easyflow中inclusive的用法示例小结

    文章介绍了在jd-easyflow中使用inclusive进行条件分支配置的方法,当conditionType设置为inclusive时,所有条件分支都会被评估,而不仅仅是一个条件满足就终止,本文给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧
    2024-11-11

最新评论