iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >Spring源码剖析1:初探Spring IOC核心流程
  • 609
分享到

Spring源码剖析1:初探Spring IOC核心流程

2023-06-02 16:06:51 609人浏览 独家记忆
摘要

本文大致地介绍了ioc容器的初始化过程,只列出了比较重要的过程和代码,可以从中看出ioC容器执行的大致流程。接下来的文章会更加深入剖析Bean容器如何解析xml,注册和初始化bean,以及如何获取bean实例等详细的过程。转自:Http:/

本文大致地介绍了ioc容器的初始化过程,只列出了比较重要的过程和代码,可以从中看出ioC容器执行的大致流程。

接下来的文章会更加深入剖析Bean容器如何解析xml,注册和初始化bean,以及如何获取bean实例等详细的过程。

转自:Http://www.importnew.com/19243.html

1\. 初始化

大致单步跟了下spring IOC的初始化过程,整个脉络很庞大,初始化的过程主要就是读取XML资源,并解析,最终注册到Bean Factory中:

[![](https://file.lsjlt.com/upload/202306/01/q33awwoghvh.jpg "flow")](https://file.lsjlt.com/upload/202306/01/q33awwoghvh.jpg "flow")

在完成初始化的过程后,Bean们就在BeanFactory中蓄势以待地等调用了。下面通过一个具体的例子,来详细地学习一下初始化过程,例如当加载下面一个bean:

```

<bean id="XiaoWang" class="com.springstudy.talentshow.SuperInstrumentalist">

    <property name="instruments">

        <list>

            <ref bean="piano"/>

            <ref bean="saxophone"/>

        </list>

    </property>

</bean>

```

加载时需要读取、解析、注册bean,这个过程具体的调用栈如下所示:

[![](https://file.lsjlt.com/upload/202306/01/ljqGovir2rf.jpg "load")](https://file.lsjlt.com/upload/202306/01/ljqgovir2rf.jpg "load")

下面对每一步的关键的代码进行详细分析:

#### 准备

保存配置位置,并刷新

在调用ClassPathXmlApplicationContext后,先会将配置位置信息保存到configLocations,供后面解析使用,之后,会调用`AbstractApplicationContext`的refresh方法进行刷新:

    

```

public ClassPathXmlApplicationContext(String[] configLocations, boolean refresh,

        ApplicationContext parent) throws BeansException {

    super(parent);

    // 保存位置信息,比如`com/springstudy/talentshow/talent-show.xml`

    setConfigLocations(configLocations);

    if (refresh) {

        // 刷新

        refresh();

    }

}

public void refresh() throws BeansException, IllegalStateException {

    synchronized (this.startupShutdownMonitor) {

        // Prepare this context for refreshing.

        prepareRefresh();

        // Tell the subclass to refresh the internal bean factory.

        ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();

        // Prepare the bean factory for use in this context.

        prepareBeanFactory(beanFactory);

        try {

            // Allows post-processing of the bean factory in context subclasses.

            postProcessBeanFactory(beanFactory);

            // Invoke factory processors reGIStered as beans in the context.

            invokeBeanFactoryPostProcessors(beanFactory);

            // Register bean processors that intercept bean creation.

            registerBeanPostProcessors(beanFactory);

            // Initialize message source for this context.

            initMessageSource();

            // Initialize event multicaster for this context.

            initApplicationEventMulticaster();

            // Initialize other special beans in specific context subclasses.

            onRefresh();

            // Check for listener beans and register them.

            registerListeners();

            // Instantiate all remaining (non-lazy-init) singletons.

            finishBeanFactoryInitialization(beanFactory);

            // Last step: publish corresponding event.

            finishRefresh();

        }

        catch (BeansException ex) {

            // Destroy already created singletons to avoid dangling resources.

            destroyBeans();

            // Reset 'active' flag.

            cancelRefresh(ex);

            // Propagate exception to caller.

            throw ex;

        }

    }

}

```

创建载入BeanFactory

```

protected final void refreshBeanFactory() throws BeansException {

    // ... ...

    DefaultListableBeanFactory beanFactory = createBeanFactory();

    // ... ...

    loadBeanDefinitions(beanFactory);

    // ... ...

}

```

创建XMLBeanDefinitionReader

```

protected void loadBeanDefinitions(DefaultListableBeanFactory beanFactory)

     throws BeansException, IOException {

    // Create a new XmlBeanDefinitionReader for the given BeanFactory.

    XmlBeanDefinitionReader beanDefinitionReader = new XmlBeanDefinitionReader(beanFactory);

    // ... ...

    // Allow a subclass to provide custom initialization of the reader,

    // then proceed with actually loading the bean definitions.

    initBeanDefinitionReader(beanDefinitionReader);

    loadBeanDefinitions(beanDefinitionReader);

```

#### 读取

创建处理每一个resource

```

public int loadBeanDefinitions(String location, Set<Resource> actualResources)

     throws BeanDefinitionStoreException {

    // ... ...

    // 通过Location来读取Resource

    Resource[] resources = ((ResourcePatternResolver) resourceLoader).getResources(location);

    int loadCount = loadBeanDefinitions(resources);

    // ... ...

}

public int loadBeanDefinitions(Resource... resources) throws BeanDefinitionStoreException {

    Assert.notNull(resources, "Resource array must not be null");

    int counter = 0;

    for (Resource resource : resources) {

        // 载入每一个resource

        counter += loadBeanDefinitions(resource);

    }

    return counter;

}

```

处理XML每个元素

```

protected void parseBeanDefinitions(Element root, BeanDefinitionParserDelegate delegate) {

    // ... ...

    nodeList nl = root.getChildNodes();

    for (int i = 0; i < nl.getLength(); i++) {

        Node node = nl.item(i);

        if (node instanceof Element) {

            Element ele = (Element) node;

            if (delegate.isDefaultNamespace(ele)) {

                // 处理每个xml中的元素,可能是import、alias、bean

                parseDefaultElement(ele, delegate);

            }

            else {

                delegate.parseCustomElement(ele);

            }

        }

    }

    // ... ...

}

```

解析和注册bean

```

protected void processBeanDefinition(Element ele, BeanDefinitionParserDelegate delegate) {

    // 解析

    BeanDefinitionHolder bdHolder = delegate.parseBeanDefinitionElement(ele);

    if (bdHolder != null) {

        bdHolder = delegate.decorateBeanDefinitionIfRequired(ele, bdHolder);

        try {

            // 注册

            // Register the final decorated instance.

            BeanDefinitionReaderUtils.registerBeanDefinition(

                bdHolder, getReaderContext().getRegistry());

        }

        catch (BeanDefinitionStoreException ex) {

            getReaderContext().error("Failed to register bean definition with name '" +

                    bdHolder.getBeanName() + "'", ele, ex);

        }

        // Send registration event.

        getReaderContext().fireComponentRegistered(new BeanComponentDefinition(bdHolder));

    }

}

```

本步骤中,通过parseBeanDefinitionElement将XML的元素解析为BeanDefinition,然后存在BeanDefinitionHolder中,然后再利用BeanDefinitionHolder将BeanDefinition注册,实质就是把BeanDefinition的实例put进BeanFactory中,和后面将详细的介绍解析和注册过程。

#### 解析

[![](https://file.lsjlt.com/upload/202306/01/e3jnyjdvw0f.jpg "process")](https://file.lsjlt.com/upload/202306/01/e3jnyjdvw0f.jpg "process")

处理每个Bean的元素

```

public AbstractBeanDefinition parseBeanDefinitionElement(

        Element ele, String beanName, BeanDefinition containingBean) {

    // ... ...

    // 创建beandefinition

    AbstractBeanDefinition bd = createBeanDefinition(className, parent);

    parseBeanDefinitionAttributes(ele, beanName, containingBean, bd);

    bd.setDescription(DomUtils.getChildElementValueByTagName(ele, DESCRIPTION_ELEMENT));

    parseMetaElements(ele, bd);

    parseLookupOverrideSubElements(ele, bd.getMethodOverrides());

    parseReplacedMethodSubElements(ele, bd.getMethodOverrides());

    // 处理“Constructor”

    parseConstructorArgElements(ele, bd);

    // 处理“Preperty”

    parsePropertyElements(ele, bd);

    parseQualifierElements(ele, bd);

    // ... ...

}

```

处理属性的值

```

public Object parsePropertyValue(Element ele, BeanDefinition bd, String propertyName) {

    String elementName = (propertyName != null) ?

                    "<property> element for property '" + propertyName + "'" :

                    "<constructor-arg> element";

    // ... ...

    if (hasRefAttribute) {

    // 处理引用

        String refName = ele.getAttribute(REF_ATTRIBUTE);

        if (!StringUtils.hasText(refName)) {

            error(elementName + " contains empty 'ref' attribute", ele);

        }

        RuntimeBeanReference ref = new RuntimeBeanReference(refName);

        ref.setSource(extractSource(ele));

        return ref;

    }

    else if (hasValueAttribute) {

    // 处理值

        TypedStringValue valueHolder = new TypedStringValue(ele.getAttribute(VALUE_ATTRIBUTE));

        valueHolder.setSource(extractSource(ele));

        return valueHolder;

    }

    else if (subElement != null) {

    // 处理子类型(比如list、map等)

        return parsePropertySubElement(subElement, bd);

    }

    // ... ...

}

```

4 注册

```

public static void registerBeanDefinition(

        BeanDefinitionHolder definitionHolder, BeanDefinitionRegistry registry)

        throws BeanDefinitionStoreException {

    // Register bean definition under primary name.

    String beanName = definitionHolder.getBeanName();

    registry.registerBeanDefinition(beanName, definitionHolder.getBeanDefinition());

    // Register aliases for bean name, if any.

    String[] aliases = definitionHolder.getAliases();

    if (aliases != null) {

        for (String alias : aliases) {

            registry.registerAlias(beanName, alias);

        }

    }

}

public void registerBeanDefinition(String beanName, BeanDefinition beanDefinition)

        throws BeanDefinitionStoreException {

    // ......

    // 将beanDefinition注册

    this.beanDefinitionMap.put(beanName, beanDefinition);

    // ......

}

```

注册过程中,最核心的一句就是:this.beanDefinitionMap.put(beanName, beanDefinition),也就是说注册的实质就是以beanName为key,以beanDefinition为value,将其put到HashMap中。

#### 注册

```

    public static void registerBeanDefinition(

        BeanDefinitionHolder definitionHolder, BeanDefinitionRegistry registry)

        throws BeanDefinitionStoreException {

    // Register bean definition under primary name.

    String beanName = definitionHolder.getBeanName();

    registry.registerBeanDefinition(beanName, definitionHolder.getBeanDefinition());

    // Register aliases for bean name, if any.

    String[] aliases = definitionHolder.getAliases();

    if (aliases != null) {

        for (String alias : aliases) {

            registry.registerAlias(beanName, alias);

        }

    }

}

public void registerBeanDefinition(String beanName, BeanDefinition beanDefinition)

        throws BeanDefinitionStoreException {

    // ......

    // 将beanDefinition注册

    this.beanDefinitionMap.put(beanName, beanDefinition);

    // ......

```

理解了以上两个过程,我们就可以自己实现一个简单的Spring框架了。于是,我根据自己的理解实现了一个简单的IOC框架Simple Spring,有兴趣可以看看。

    

注册过程中,最核心的一句就是:`this.beanDefinitionMap.put(beanName, beanDefinition)`,也就是说注册的实质就是以beanName为key,以beanDefinition为value,将其put到HashMap中。

### 注入依赖

当完成初始化IOC容器后,如果bean没有设置lazy-init(延迟加载)属性,那么bean的实例就会在初始化IOC完成之后,及时地进行初始化。初始化时会先建立实例,然后根据配置利用反射对实例进行进一步操作,具体流程如下所示:

[![](https://file.lsjlt.com/upload/202306/01/nswbubr31eo.jpg "bean_flow")](https://file.lsjlt.com/upload/202306/01/nswbubr31eo.jpg "bean_flow")

创建bean的实例

创建bean的实例过程函数调用栈如下所示:

[![](https://file.lsjlt.com/upload/202306/01/ecbyyeldxxi.jpg "create_bean")](https://file.lsjlt.com/upload/202306/01/ecbyyeldxxi.jpg "create_bean")

注入bean的属性

注入bean的属性过程函数调用栈如下所示:

[![](https://file.lsjlt.com/upload/202306/01/wxf11bblfym.jpg "inject_property")](https://file.lsjlt.com/upload/202306/01/wxf11bblfym.jpg "inject_property")

在创建bean和注入bean的属性时,都是在doCreateBean函数中进行的,我们重点看下:

```

protected Object doCreateBean(final String beanName, final RootBeanDefinition mbd,

            final Object[] args) {

        // Instantiate the bean.

        BeanWrapper instanceWrapper = null;

        if (mbd.isSingleton()) {

            instanceWrapper = this.factoryBeanInstanceCache.remove(beanName);

        }

        if (instanceWrapper == null) {

            // 创建bean的实例

            instanceWrapper = createBeanInstance(beanName, mbd, args);

        }

     

        // ... ...

     

        // Initialize the bean instance.

        Object exposedObject = bean;

        try {

            // 初始化bean的实例,如注入属性

            populateBean(beanName, mbd, instanceWrapper);

            if (exposedObject != null) {

                exposedObject = initializeBean(beanName, exposedObject, mbd);

            }

        }

     

        // ... ...

    }

```

理解了以上两个过程,我们就可以自己实现一个简单的Spring框架了。于是,我根据自己的理解实现了一个简单的IOC框架[Simple Spring](https://GitHub.com/Yikun/simple-spring),有兴趣可以看看。

--结束END--

本文标题: Spring源码剖析1:初探Spring IOC核心流程

本文链接: https://www.lsjlt.com/news/230621.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • Spring源码剖析1:初探Spring IOC核心流程
    本文大致地介绍了IOC容器的初始化过程,只列出了比较重要的过程和代码,可以从中看出IOC容器执行的大致流程。接下来的文章会更加深入剖析Bean容器如何解析xml,注册和初始化bean,以及如何获取bean实例等详细的过程。转自:http:/...
    99+
    2023-06-02
  • Spring IOC核心流程是什么
    本篇内容主要讲解“Spring IOC核心流程是什么”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Spring IOC核心流程是什么”吧! 初始化大致单步跟了下Spring IOC的初始化过程,...
    99+
    2023-06-02
  • Spring源码剖析3:Spring IOC容器的加载过程
    本文转自五月的仓颉 https://www.cnblogs.com/xrq730本系列文章将整理到我在GitHub上的《Java面试指南》仓库,更多精彩内容请到我的仓库里查看https://github.com/h3pl/Java-Tuto...
    99+
    2023-06-02
  • Spring AOP核心功能源码分析
    这篇“Spring AOP核心功能源码分析”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Spring A...
    99+
    2023-07-05
  • Spring IOC源码剖析_如何整体认知Spring体系结构
    目录如何整体认知Spring体系结构一、来自官网的Spring二、Spring的优缺点三、一张图理解Spring Framework 4.x四、详解“七层”宝塔1. 核心容器(Cor...
    99+
    2024-04-02
  • MyBatis核心源码深度剖析SQL语句执行过程
    目录1 SQL语句的执行过程介绍2 SQL执行的入口分析2.1 为Mapper接口创建代理对象2.2 执行代理逻辑3 查询语句的执行过程分析3.1 selectOne方法分...
    99+
    2024-04-02
  • Spring源码分析容器启动流程
    目录前言源码解析1、初始化流程流程分析核心代码剖析2、刷新流程流程分析核心代码剖析前言 本文基于 Spring 的 5.1.6.RELEASE 版本 Spring的启动流程可以归纳为...
    99+
    2024-04-02
  • Spring Bean的实例化之属性注入源码剖析过程
    前言 这一章节我们来讨论创建Bean过程中的属性注入,在Spring的IOC容器启动过程中,会把定义的Bean封装成BeanDefinition注册到一个ConcurrentHash...
    99+
    2024-04-02
  • spring boot 加载web容器tomcat流程源码分析
    我本地的springboot版本是2.5.1,后面的分析都是基于这个版本 <parent> <groupId>org.springfra...
    99+
    2024-04-02
  • Spring Boot源码解读与原理剖析:深入探索Java开发的奥秘!
    评论区留言赠书15本 关注+点赞+评论,评论区回复“Spring Boot源码解读与原理剖析:深入探索Java开发的奥秘!” 每篇最多评论3条!!采用抽奖助手自动拉取评论区有效评论送书两本, 开奖时间:9月11号 承载着作者的厚...
    99+
    2023-09-13
    Java SpringBoot 原力计划
  • Spring源码剖析3:懒加载的单例Bean获取过程分析
    spring ioc 容器的加载流程...
    99+
    2023-06-02
  • Spring源码解析之循环依赖的实现流程
    目录前言循环依赖实现流程前言 上篇文章中我们分析完了Spring中Bean的实例化过程,但是没有对循环依赖的问题进行分析,这篇文章中我们来看一下spring是如何解决循环依赖的实现。...
    99+
    2024-04-02
  • Spring Security过滤器链加载执行流程源码解析
    目录Spring Security实现原理一、Spring Security过滤器链加载1、注册名为 springSecurityFilterChain的过滤器2、查看 Delega...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作