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注解内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Spring Boot Admin监控服务如何使用

    Spring Boot Admin监控服务如何使用

    这篇文章主要介绍了Spring Boot Admin监控服务如何使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-04-04
  • java 反射 动态调用不同类的静态方法(推荐)

    java 反射 动态调用不同类的静态方法(推荐)

    下面小编就为大家带来一篇JAVA 反射 动态调用不同类的静态方法(推荐)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-08-08
  • SpringBoot实现文件上传与下载功能的示例代码

    SpringBoot实现文件上传与下载功能的示例代码

    文件上传与下载是Web应用开发中常用的功能之一。接下来我们将讨论如何在Spring Boot的Web应用开发中,如何实现文件的上传与下载,感兴趣的可以了解一下
    2022-06-06
  • MyBatis 自带连接池的具体实现

    MyBatis 自带连接池的具体实现

    MyBatis自带的PooledDataSource实现了一个简单的数据库连接池,适合简单场景,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2024-12-12
  • Java最简单的DES加密算法实现案例

    Java最简单的DES加密算法实现案例

    下面小编就为大家带来一篇Java最简单的DES加密算法实现案例。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-06-06
  • Java中HashMap的常见用法详解

    Java中HashMap的常见用法详解

    这篇文章主要介绍了Java中HashMap的常见用法详解,HashMap是Java中的一个常用子类,它是java.util.HashMap<k,v>集合,实现了Map<k,v>接口, HashMap可以存储键值对,通过键来快速访问值,在HashMap中,键是唯一的,而值可以重复,需要的朋友可以参考下
    2023-09-09
  • JAVACORE与HEAPDUMP生成方法

    JAVACORE与HEAPDUMP生成方法

    JavaCore文件主要保存的是Java应用各线程在某一时刻的运行的位置,即JVM执行到哪一个类、哪一个方法、哪一个行上,它是一个文本文件,打开后可以看到每一个线程的执行栈,以stack trace的显示,本文介绍JAVACORE与HEAPDUMP生成大法,感兴趣的朋友跟随小编一起看看吧
    2024-08-08
  • spring boot实现阿里云视频点播上传视频功能(复制粘贴即可)

    spring boot实现阿里云视频点播上传视频功能(复制粘贴即可)

    这篇文章主要介绍了spring boot实现阿里云视频点播上传视频功能(复制粘贴即可),本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-12-12
  • springboot中在非bean类中调用bean的实现方法

    springboot中在非bean类中调用bean的实现方法

    在Spring Boot中,非Bean类调用Bean方法通常需要通过静态方法获取Bean实例,然后调用相应的方法,这种方法避免了直接在非Bean类中注入Bean,保持了代码的简洁和可维护性,通过这种方式,可以在不改变原有代码结构的情况下,实现Bean方法的调用
    2025-02-02
  • Java中的javaBean、vo、entity、domain和pojo

    Java中的javaBean、vo、entity、domain和pojo

    这篇文章主要介绍了Java中的javaBean、vo、entity、domain和pojo用法,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-12-12

最新评论