Spring源码解析——refresh方法

refresh方法在ConfigurableApplicationContext类中定义的具体实现是在AbstractApplicationContext中实现。这个方法的原文描述信息是:由于这是一个启动方法,如果它调用失败,它应该销毁已创建的单例,以避免悬空资源。换句话说,在调用该方法之后,应该实例化所有单例或者不是单例的对象。

接下来分析refresh方法

public void refresh() throws BeansException, IllegalStateException {
        synchronized (this.startupShutdownMonitor) {
//----------------------------------------1--------------------------------------//
            // 准备刷新和加载之前的上下文
            prepareRefresh();
//----------------------------------------1--------------------------------------//

//----------------------------------------2--------------------------------------//
            //告诉子类刷新内部bean工厂
            ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();
//----------------------------------------2--------------------------------------//

//----------------------------------------3--------------------------------------//
            // 准备在此上下文中使用的bean工厂
            prepareBeanFactory(beanFactory);
//----------------------------------------3--------------------------------------//

            try {
        //----------------------------------------4--------------------------------------//
                //注册对应的BeanPostProcessor接口,就是ServletContextAwareProcessor
                postProcessBeanFactory(beanFactory);
      //----------------------------------------4--------------------------------------//

  //----------------------------------------5--------------------------------------//
                //调用注册的BeanFactoryPostProcessor的postProcessBeanDefinitionRegistry方法
                invokeBeanFactoryPostProcessors(beanFactory);
  //----------------------------------------5--------------------------------------//

//----------------------------------------6--------------------------------------//
                // 注册拦截bean创建的bean处理器
                registerBeanPostProcessors(beanFactory);
//----------------------------------------6--------------------------------------//

                //初始化上下文的消息源。
                initMessageSource();
                //初始化上下文的事件传播器。
                initApplicationEventMulticaster();
                //在特定上下文子类中初始化其他特殊bean。
                onRefresh();
                // 检查监听器bean并注册它们
                registerListeners();
//----------------------------------------7--------------------------------------//
                //实例化所有剩余(非延迟初始化)单例。
                finishBeanFactoryInitialization(beanFactory);
//----------------------------------------7--------------------------------------//

//----------------------------------------8--------------------------------------//
                //最后一步:发布相应的事件。
                finishRefresh();
//----------------------------------------8-------------------------------------//
            }

            catch (BeansException ex) {
                if (logger.isWarnEnabled()) {
                    logger.warn("Exception encountered during context initialization - " +
                            "cancelling refresh attempt: " + ex);
                }

                // Destroy already created singletons to avoid dangling resources.
                destroyBeans();

                // Reset 'active' flag.
                cancelRefresh(ex);

                // Propagate exception to caller.
                throw ex;
            }

            finally {
                // Reset common introspection caches in Spring's core, since we
                // might not ever need metadata for singleton beans anymore...
                resetCommonCaches();
            }
        }
    }

方法1:prepareRefresh

protected void prepareRefresh() {
        this.startupDate = System.currentTimeMillis();
        this.closed.set(false);
        this.active.set(true);
        if (logger.isDebugEnabled()) {
            if (logger.isTraceEnabled()) {
                logger.trace("Refreshing " + this);
            }
            else {
                logger.debug("Refreshing " + getDisplayName());
            }
        }
        //在上下文环境中初始化任何占位符属性源
        initPropertySources();
        // 验证指定的每个属性是不是可以解析的
        getEnvironment().validateRequiredProperties();
        // 初始化传播器
        this.earlyApplicationEvents = new LinkedHashSet<>();
    }

方法2:obtainFreshBeanFactory

    protected ConfigurableListableBeanFactory obtainFreshBeanFactory() {
        //初始化BeanFactory
        refreshBeanFactory();
        //返回实例化的BeanFactory
        return getBeanFactory();
    }

其中的refreshBeanFactory方法的作用用来实例化BeanFactory并加载解析对应的Bean配置,其加载Bean的过程和前面讲到的BeanFactory加载的过程相同

protected final void refreshBeanFactory() throws BeansException {
        //通过beanFactoryMonitor这个Bean工厂的监视器来判断是否已经实例化了BeanFactory
        if (hasBeanFactory()) {
            //如果实例化了BeanFactory,则先销毁所有的单例Bean
            destroyBeans();
            //关闭BeanFactory,也就是将beanFactory的值置为null
            closeBeanFactory();
        }
        try {
            //创建BeanFactory实例,如果实现了ConfigurableApplicationContext则返回父上下文的内部bean工厂;否则,返回父上下文本身
            DefaultListableBeanFactory beanFactory = createBeanFactory();
            beanFactory.setSerializationId(getId());
            //如果用户自定义了允许Bean定义覆盖和允许循环引用则设置为定义的,默认为不允许
            customizeBeanFactory(beanFactory);
            //加载bean,前面已经讲过bean的加载
            loadBeanDefinitions(beanFactory);
            synchronized (this.beanFactoryMonitor) {
                this.beanFactory = beanFactory;
            }
        }
        catch (IOException ex) {
            throw new ApplicationContextException("I/O error parsing bean definition source for " + getDisplayName(), ex);
        }
    }

方法3:prepareBeanFactory

这个方法主要是配置BeanFactory的各种类加载器,需要的依赖和需要忽略的依赖,后处理器,解析器等

protected void prepareBeanFactory(ConfigurableListableBeanFactory beanFactory) {
        //设置加载Bean用的类加载器
        beanFactory.setBeanClassLoader(getClassLoader());
        //设置Bean中表达式解析用的解析器,默认的SpEL解析器
        beanFactory.setBeanExpressionResolver(new StandardBeanExpressionResolver(beanFactory.getBeanClassLoader()));
        //添加属性注册时候的注册器,可以自定义注册器,只需要实现PropertyEditorRegistrar接口
        beanFactory.addPropertyEditorRegistrar(new ResourceEditorRegistrar(this, getEnvironment()));
        //添加ApplicationContextAwareProcessor(其中包含了ApplicationContext,起到了传递的作用)
        beanFactory.addBeanPostProcessor(new ApplicationContextAwareProcessor(this));
        //忽略给定的自动装配依赖关系接口,放到ignoredDependencyInterfaces中保存
        beanFactory.ignoreDependencyInterface(EnvironmentAware.class);
        beanFactory.ignoreDependencyInterface(EmbeddedValueResolverAware.class);
        beanFactory.ignoreDependencyInterface(ResourceLoaderAware.class);
        beanFactory.ignoreDependencyInterface(ApplicationEventPublisherAware.class);
        beanFactory.ignoreDependencyInterface(MessageSourceAware.class);
        beanFactory.ignoreDependencyInterface(ApplicationContextAware.class);
        // BeanFactory interface not registered as resolvable type in a plain factory.
        // MessageSource registered (and found for autowiring) as a bean.
        //自动装配值注册特殊依赖关系类型,放到resolvableDependencies集合中保存
        beanFactory.registerResolvableDependency(BeanFactory.class, beanFactory);
        beanFactory.registerResolvableDependency(ResourceLoader.class, this);
        beanFactory.registerResolvableDependency(ApplicationEventPublisher.class, this);
        beanFactory.registerResolvableDependency(ApplicationContext.class, this);
        //注册ApplicationListenerDetector作为早期后处理器以检测内部bean实现了ApplicationListener的情况
        // Register early post-processor for detecting inner beans as ApplicationListeners.
        beanFactory.addBeanPostProcessor(new ApplicationListenerDetector(this));
        // Detect a LoadTimeWeaver and prepare for weaving, if found.
        //如果beanFactory中包含loadTimeWeaver名称的bean,则需要使用对应的临时的类加载器处理
        if (beanFactory.containsBean(LOAD_TIME_WEAVER_BEAN_NAME)) {
            beanFactory.addBeanPostProcessor(new LoadTimeWeaverAwareProcessor(beanFactory));
            // Set a temporary ClassLoader for type matching.
            beanFactory.setTempClassLoader(new ContextTypeMatchClassLoader(beanFactory.getBeanClassLoader()));
        }
        // Register default environment beans.
        //注册默认的需要的上下文环境bean,包含environment,systemProperties和systemEnvironment
        if (!beanFactory.containsLocalBean(ENVIRONMENT_BEAN_NAME)) {
            beanFactory.registerSingleton(ENVIRONMENT_BEAN_NAME, getEnvironment());
        }
        if (!beanFactory.containsLocalBean(SYSTEM_PROPERTIES_BEAN_NAME)) {
            beanFactory.registerSingleton(SYSTEM_PROPERTIES_BEAN_NAME, getEnvironment().getSystemProperties());
        }
        if (!beanFactory.containsLocalBean(SYSTEM_ENVIRONMENT_BEAN_NAME)) {
            beanFactory.registerSingleton(SYSTEM_ENVIRONMENT_BEAN_NAME, getEnvironment().getSystemEnvironment());
        }
    }

方法4:postProcessBeanFactory

这个方法主要注册web请求相关的处理器和bean以及配置

protected void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) {
        //将ServletContextAwareProcessor注册到bean注册前处理器中
        beanFactory.addBeanPostProcessor(new ServletContextAwareProcessor(this.servletContext, this.servletConfig));
        //将ServletContextAware和ServletConfigAware设置为忽略依赖
        beanFactory.ignoreDependencyInterface(ServletContextAware.class);
        beanFactory.ignoreDependencyInterface(ServletConfigAware.class);
        //注册特定于Web的范围(如request,session和application),添加对应的web请求依赖(ServletRequest,ServletResponse,HttpSession,WebRequest使用各自对应的BeanFactory)
        WebApplicationContextUtils.registerWebApplicationScopes(beanFactory, this.servletContext);
        //注册环境变量,在servletContext和servletConfig中设置的和自带的
        WebApplicationContextUtils.registerEnvironmentBeans(beanFactory, this.servletContext, this.servletConfig);
    }

方法5:invokeBeanFactoryPostProcessors

这个方法主要是调用实现了的bean后处理器

protected void invokeBeanFactoryPostProcessors(ConfigurableListableBeanFactory beanFactory) {
        /**
         * 这里的调用所有的实现了BeanFactoryPostProcessor接口的类的方法,调用postProcessBeanDefinitionRegistry方法的时候按照会按照一定顺序来调用
         * 1.调用实现PriorityOrdered的BeanDefinitionRegistryPostProcessors
         * 2.调用实现Ordered的BeanDefinitionRegistryPostProcessors
         * 3.调用所有其他BeanDefinitionRegistryPostProcessors,直到不再出现其他BeanDefinitionRegistryPostProcessors
         * 调用postProcessBeanFactory没有顺序
         */
        PostProcessorRegistrationDelegate.invokeBeanFactoryPostProcessors(beanFactory, getBeanFactoryPostProcessors());
        //这里特殊处理前面说到的bean名称为loadTimeWeaver的Bean
        if (beanFactory.getTempClassLoader() == null && beanFactory.containsBean(LOAD_TIME_WEAVER_BEAN_NAME)) {
            beanFactory.addBeanPostProcessor(new LoadTimeWeaverAwareProcessor(beanFactory));
            beanFactory.setTempClassLoader(new ContextTypeMatchClassLoader(beanFactory.getBeanClassLoader()));
        }
    }

方法6:registerBeanPostProcessors

这个方法主要是注册实现了的bean后处理器

protected void registerBeanPostProcessors(ConfigurableListableBeanFactory beanFactory) {
        /**
         * 这里的注册所有的实现了BeanPostProcessor接口的类的方法一定顺序来注册
         * 1.注册实现PriorityOrdered的BeanPostProcessor
         * 2.注册实现Ordered的BeanPostProcessor
         * 3.注册所有其他BeanPostProcessor,直到不再出现其他BeanPostProcessor
         * 最后,重新注册ApplicationListenerDetector为后处理器以检测内部bean实现ApplicationListener接口的,并将其移动到处理器链的末尾
         */
        PostProcessorRegistrationDelegate.registerBeanPostProcessors(beanFactory, this);
}

方法7:finishBeanFactoryInitialization

 在这里加载并实例化所有的懒加载的bean

protected void  finishBeanFactoryInitialization(ConfigurableListableBeanFactory beanFactory) {
        //设置对应的用于类型转换的服务接口,ConversionService可以自己实现
        if (beanFactory.containsBean(CONVERSION_SERVICE_BEAN_NAME) &&
                beanFactory.isTypeMatch(CONVERSION_SERVICE_BEAN_NAME, ConversionService.class)) {
            beanFactory.setConversionService(
                    beanFactory.getBean(CONVERSION_SERVICE_BEAN_NAME, ConversionService.class));
        }

        /* Register a default embedded value resolver if no bean post-processor
         (such as a PropertyPlaceholderConfigurer bean) registered any before:
         at this point, primarily for resolution in annotation attribute values.*/
        if (!beanFactory.hasEmbeddedValueResolver()) {
            beanFactory.addEmbeddedValueResolver(strVal -> getEnvironment().resolvePlaceholders(strVal));
        }

        // Initialize LoadTimeWeaverAware beans early to allow for registering their transformers early.
        String[] weaverAwareNames = beanFactory.getBeanNamesForType(LoadTimeWeaverAware.class, false, false);
        for (String weaverAwareName : weaverAwareNames) {
            getBean(weaverAwareName);
        }

        //停止使用临时ClassLoader进行类型匹配,因为已经加载完了bean
        beanFactory.setTempClassLoader(null);

        //冻结所以的bean定义,说明注册的bean定义将不被修改或任何进一步的处理
        beanFactory.freezeConfiguration();

        //初始化剩下的单实例(非惰性),在这个方法中最终会调用getBean方法,就会实例化bean
        beanFactory.preInstantiateSingletons();
}

方法8:finishRefresh

protected void finishRefresh() {
                //清除上下文级资源缓存
        clearResourceCaches();

        // 为此上下文初始化生命周期处理器,可以自定义LifecycleProcessor,如果没有则使用默认的DefaultLifecycleProcessor
        initLifecycleProcessor();

        // 刷新bean的生命周期  
        getLifecycleProcessor().onRefresh();

        // Publish the final event.
        publishEvent(new ContextRefreshedEvent(this));

        // Participate in LiveBeansView MBean, if active.
        LiveBeansView.registerApplicationContext(this);
}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
`refresh()` 是 Spring 框架中的一个方法,它用于刷新应用程序上下文(ApplicationContext)以更新其内部状态。在 Spring 框架中,`refresh()` 方法是非常重要的,因为它负责完成应用程序上下文的初始化和配置,并准备好所有的单例 bean 以供使用。 下面是 `refresh()` 方法的主要流程: 1. 准备刷新过程中需要用到的变量和标志位; 2. 调用 `prepareRefresh()` 方法,进行一些预处理工作; 3. 调用 `obtainFreshBeanFactory()` 方法,创建 BeanFactory 并进行一些初始化工作; 4. 调用 `prepareBeanFactory(beanFactory)` 方法,对 BeanFactory 进行一些后续处理; 5. 调用 `postProcessBeanFactory(beanFactory)` 方法,对 BeanFactory 进行后置处理; 6. 调用 `invokeBeanFactoryPostProcessors(beanFactory)` 方法,执行 BeanFactoryPostProcessor 的 postProcessBeanFactory() 方法; 7. 调用 `registerBeanPostProcessors(beanFactory)` 方法,注册 BeanPostProcessor 实例; 8. 调用 `initMessageSource()` 方法,初始化 MessageSource 组件; 9. 调用 `initApplicationEventMulticaster()` 方法,初始化 ApplicationEventMulticaster 组件; 10. 调用 `onRefresh()` 方法,进行一些自定义的刷新工作; 11. 调用 `registerListeners()` 方法,注册事件监听器; 12. 调用 `finishBeanFactoryInitialization(beanFactory)` 方法,完成所有非延迟初始化的单例 bean 的初始化工作; 13. 调用 `finishRefresh()` 方法,完成上下文的刷新工作。 需要注意的是,`refresh()` 方法在执行过程中会涉及到很多细节,比如如何处理环境变量、如何处理自定义的 bean 定义、如何处理多个上下文之间的关系等等。如果需要深入了解 `refresh()` 方法的实现细节,可以查看 Spring 框架的代码。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值