Spring Bean注册的源码解析

在这里插入图片描述

概要:

我们都知道Spring的BeanFactory是管理Bean,所以我们只要关注于BeanFactrory就能找到关于Bean的注册地方

Spring 加载Bean 分为3个步骤:

  • 定位
    • 主要是用来定位资源信息(需要加载的Bean所在位置)
  • 加载
    • 找到需要加载的bean,解析他的参数(构造参数,初始化方法,等等)
  • 注册
    • 把加载好的Bean保存到IOC容器中

源码解析

这里我们先使用配置文件的形式看Spring源码

我们先找到spring的切入类org.springframework.context.support.ClassPathXmlApplicationContext

// 创建一个新的ClassPathXmlApplicationContext,从给定的XML文件加载定义并自动刷新上下文。
public ClassPathXmlApplicationContext(String configLocation) throws BeansException {
    this(new String[] {configLocation}, true, null);
}

调用了自己的其他构造方法

//使用父级创建新的ClassPathXmlApplicationContext,并且从给的XML文件中加载定义。
//configLocations :传入的配置文件的位置
//refresh:是否自动刷新 
//parent: 准备当父级上下文的对象
public ClassPathXmlApplicationContext(
    String[] configLocations, boolean refresh, @Nullable ApplicationContext parent)
    throws BeansException {
    //设置配置信息到ClassPathXmlApplication中 
    super(parent);
    //设置配置文件地址路径
    setConfigLocations(configLocations);
    if (refresh) {//判断是否刷新上下文
        //刷新上下文  核心方法
        refresh();
    }
}

解析 super(parent);`

一直点父类的构造函数,会发现 这个只是创建一个org.springframework.context.support.AbstractApplicationContext#AbstractApplicationContext() ,并且包含一个resourcePatternResolver

//1.创建AbstractApplicationContext 对象,
//2. 并且把传入的applicationContext设置为父上下文
public AbstractApplicationContext(@Nullable ApplicationContext parent) {
    this();
    setParent(parent);
}

查看this()构造函数

//org.springframework.context.support.AbstractApplicationContext#AbstractApplicationContext()
public AbstractApplicationContext() {
    this.resourcePatternResolver = getResourcePatternResolver();
}

解析 setConfigLocations(configLocations);

传入的配置文件地址进行转换保存到configLocations属性中

//设置此应用程序上下文的配置位置。如果未设置,则设置为null
public void setConfigLocations(@Nullable String... locations) {
    if (locations != null) {
        Assert.noNullElements(locations, "Config locations must not be null");
        this.configLocations = new String[locations.length];
        for (int i = 0; i < locations.length; i++) {
            //resolvePath():解析给定路径,必要时用相应的环境属性值替换占位符。应用于配置位置。
            this.configLocations[i] = resolvePath(locations[i]).trim();
        }
    }
    else {
        this.configLocations = null;
    }
}

重要解析 refresh();

这是重要的方法。我们先来看下这个方法内容

public void refresh() throws BeansException, IllegalStateException {
    synchronized (this.startupShutdownMonitor) {
       
        ...

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

        ...

    }
}

因为现在我们只看加载Bean的源码。,所以其他的我就先忽略了

解析 obtainFreshBeanFactory();

//只有2个方法
protected ConfigurableListableBeanFactory obtainFreshBeanFactory() {
    //刷新Bean工厂
    refreshBeanFactory();
    //返回
    return getBeanFactory();
}

这里发现,只有2行代码,刷新Bean工厂和返回Bean工厂,那么我们先看刷新Bean工厂的方法

解析 refreshBeanFactory()

我们看refreshBeanFactory()这个方法的时候,会发现在org.springframework.context.support.AbstractApplicationContext 类中是抽象方法

这里spring用了模板方法,所以我们只要找到实现的子类就可以了org.springframework.context.support.AbstractRefreshableApplicationContext

接下来我们来看重写后的方法refreshBeanFactory()

@Override
protected final void refreshBeanFactory() throws BeansException {
    //判断是否有Bean工厂已经存在
    if (hasBeanFactory()) {//beanFactory!=null
        //把注册的beans都清空掉
        destroyBeans();
        //关闭
        closeBeanFactory();
    }
    try {
        //创建bean工厂   如果实现了ConfigurableBeanFactory 则返回实现后的上一级BeanFactory ,否则返回当前的BeanFactory
        DefaultListableBeanFactory beanFactory = createBeanFactory();
        //设置序列化ID 用来序列化
        beanFactory.setSerializationId(getId());
        //自定义此上下文使用的内部bean工厂。
        customizeBeanFactory(beanFactory);
        //加载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);
    }
}

解析 org.springframework.context.support.AbstractRefreshableApplicationContext#hasBeanFactory

主要看benFactory是否已经存在了。

protected final boolean hasBeanFactory() {
    synchronized (this.beanFactoryMonitor) {
        return (this.beanFactory != null);
    }
}

如果存在话,则需要调用

org.springframework.context.support.AbstractApplicationContext#destroyBeans 清空BeanFactory

org.springframework.context.support.AbstractRefreshableApplicationContext#closeBeanFactory 关掉beanFactory

解析 org.springframework.context.support.AbstractRefreshableApplicationContext#createBeanFactory

protected DefaultListableBeanFactory createBeanFactory() {
    //创建一个 DefaultListableBeanFactory
    //getInternalParentBeanFactory() 如果它实现了ConfigurableApplicationContext,则返回父上下文的内部bean工厂;否则,返回父级上下文本身。
    return new DefaultListableBeanFactory(getInternalParentBeanFactory());
}

这里我们发现返回的是一个org.springframework.beans.factory.support.DefaultListableBeanFactory

后面这2步主要为beanFactory设置参数

//设置序列化ID 用来序列化
beanFactory.setSerializationId(getId());
//自定义此上下文使用的内部bean工厂。
customizeBeanFactory(beanFactory);

解析 org.springframework.context.support.AbstractRefreshableApplicationContext#loadBeanDefinitions

同样这里也是模板方法,所以我们只要找到对应的实现子类,这里我们是org.springframework.context.support.AbstractXmlApplicationContext

org.springframework.context.support.AbstractXmlApplicationContext#loadBeanDefinitions(org.springframework.beans.factory.support.DefaultListableBeanFactory)

protected void loadBeanDefinitions(DefaultListableBeanFactory beanFactory) throws BeansException, IOException {
    // 给BeanFactory创建一个新的XmlBeanDefinitionReader。  定位XML的位置
    XmlBeanDefinitionReader beanDefinitionReader = new XmlBeanDefinitionReader(beanFactory);

    //使用此上下文的资源加载环境配置bean定义读取器。
    beanDefinitionReader.setEnvironment(this.getEnvironment());
    //设置资源地址
    beanDefinitionReader.setResourceLoader(this);
    //设置资源对象转换器
    beanDefinitionReader.setEntityResolver(new ResourceEntityResolver(this));

    //设置是否对XML进行校验
    initBeanDefinitionReader(beanDefinitionReader);

    //开始加载bean定义
    loadBeanDefinitions(beanDefinitionReader);
}

上面都是初始化的步骤

  • 设置环境变量
  • 设置资源文件
  • 设置资源对象转换器,用来转换
  • 设置是否对XML进行校验

解析 org.springframework.context.support.AbstractXmlApplicationContext#loadBeanDefinitions(org.springframework.beans.factory.xml.XmlBeanDefinitionReader)

这里有2种读取方式。一种是根据Resource资源文件,一种是根据指定路径

protected void loadBeanDefinitions(XmlBeanDefinitionReader reader) throws BeansException, IOException {
    //获取配置资源文件 class-*.xml
    Resource[] configResources = getConfigResources();
    if (configResources != null) {
        //进行加载
        reader.loadBeanDefinitions(configResources);
    }
    //获取路径
    String[] configLocations = getConfigLocations();
    if (configLocations != null) {
        reader.loadBeanDefinitions(configLocations);
    }
}

我们先来看Resource的参数的构造方法

主要是 把传入的集合进行循环,单独执行

public int loadBeanDefinitions(Resource... resources) throws BeanDefinitionStoreException {
    Assert.notNull(resources, "Resource array must not be null");
    int count = 0;
    //循环资源
    for (Resource resource : resources) {
        count += loadBeanDefinitions(resource);
    }
    return count;
}

解析 org.springframework.beans.factory.support.BeanDefinitionReader#loadBeanDefinitions(org.springframework.core.io.Resource)

这里同样要找到子类的实现org.springframework.beans.factory.xml.XmlBeanDefinitionReader#loadBeanDefinitions(org.springframework.core.io.Resource)

//从指定的XML文件读取Bean定义
public int loadBeanDefinitions(Resource resource) throws BeanDefinitionStoreException {
    //new EncodedResource(resource) 对文件进行编码设置
    return loadBeanDefinitions(new EncodedResource(resource));
}

解析 org.springframework.beans.factory.xml.XmlBeanDefinitionReader#loadBeanDefinitions(org.springframework.core.io.support.EncodedResource)

public int loadBeanDefinitions(EncodedResource encodedResource) throws BeanDefinitionStoreException {
   ...
    //获取当前正在加载的资源文件
    Set<EncodedResource> currentResources = this.resourcesCurrentlyBeingLoaded.get();
    if (currentResources == null) {
        //创建一个
        currentResources = new HashSet<>(4);
        //为空的话 设置一个
        this.resourcesCurrentlyBeingLoaded.set(currentResources);
    }
    if (!currentResources.add(encodedResource)) {
        throw new BeanDefinitionStoreException(
            "Detected cyclic loading of " + encodedResource + " - check your import definitions!");
    }
    try {
        InputStream inputStream = encodedResource.getResource().getInputStream();
        try {
            InputSource inputSource = new InputSource(inputStream);
            if (encodedResource.getEncoding() != null) {//设置编码
                inputSource.setEncoding(encodedResource.getEncoding());
            }
            //do 开始真正 的工作  开始i执行 加载的步骤
            return doLoadBeanDefinitions(inputSource, encodedResource.getResource());
        }
        finally {
            inputStream.close();
        }
    }
    catch (IOException ex) {
        throw new BeanDefinitionStoreException(
            "IOException parsing XML document from " + encodedResource.getResource(), ex);
    }
    finally {
        //从threadload中删除当前对象
        currentResources.remove(encodedResource);
        if (currentResources.isEmpty()) {
            this.resourcesCurrentlyBeingLoaded.remove();
        }
    }
}

代码可以看到

  • 获取加载的资源
  • 对文件进行设置编码
  • doLoadBeanDefinitions()

下面我们

解析 org.springframework.beans.factory.xml.XmlBeanDefinitionReader#doLoadBeanDefinitions

protected int doLoadBeanDefinitions(InputSource inputSource, Resource resource)
    throws BeanDefinitionStoreException {

    try {
        //解析获得Doucement对象 
        Document doc = doLoadDocument(inputSource, resource);
        //注册DOM文件中包含的bean定义。返回注册的bean个数 
        int count = registerBeanDefinitions(doc, resource);
        if (logger.isDebugEnabled()) {
            logger.debug("Loaded " + count + " bean definitions from " + resource);
        }
        return count;
   ...
}
  • 通过DocumentBuilder 获取Document 对象
  • 注册DOM文件中包含的bean定义。返回注册的bean个数

解析 org.springframework.beans.factory.xml.XmlBeanDefinitionReader#registerBeanDefinitions

public int registerBeanDefinitions(Document doc, Resource resource) throws BeanDefinitionStoreException {
    //创建BeanDefinitionDocumentReader
    BeanDefinitionDocumentReader documentReader = createBeanDefinitionDocumentReader();
    //获取注册前的个数 beanDefinitionMap 中的个数
    int countBefore = getRegistry().getBeanDefinitionCount();

    //开始注册
    documentReader.registerBeanDefinitions(doc, createReaderContext(resource));

    return getRegistry().getBeanDefinitionCount() - countBefore;
}
  • 通过反射获取DefaultBeanDefinitionDocumentReader 对象
  • 获取注册前的IOC中的个数
  • 开始注册

解析 org.springframework.beans.factory.xml.DefaultBeanDefinitionDocumentReader#registerBeanDefinitions

public void registerBeanDefinitions(Document doc, XmlReaderContext readerContext) {
    this.readerContext = readerContext;
    //开始注册
    doRegisterBeanDefinitions(doc.getDocumentElement());
}
  • 设置readerContext
  • 开始注册操作(还不是真正的注册,只是解析操作)

解析 org.springframework.beans.factory.xml.DefaultBeanDefinitionDocumentReader#doRegisterBeanDefinitions

protected void doRegisterBeanDefinitions(Element root) {
    //任何嵌套的<beans>元素都将导致此方法中的递归。
    // 为了正确传播和保留<beans> default- *属性,请跟踪当前(父)委托,该委托可以为null。
    // 创建新的(子)委托,引用父项以进行回退,然后最终将this.delegate重置为其原始(父)引用。
    //此行为模拟了一堆代理,而实际上并不需要一个代理。
    BeanDefinitionParserDelegate parent = this.delegate;
    //创建一个BeanDefinitionParserDelegat委派者
    this.delegate = createDelegate(getReaderContext(), root, parent);

    if (this.delegate.isDefaultNamespace(root)) {
        //获取 profile
        String profileSpec = root.getAttribute(PROFILE_ATTRIBUTE);
        if (StringUtils.hasText(profileSpec)) {
            //,;
            String[] specifiedProfiles = StringUtils.tokenizeToStringArray(
                profileSpec, BeanDefinitionParserDelegate.MULTI_VALUE_ATTRIBUTE_DELIMITERS);
            // We cannot use Profiles.of(...) since profile expressions are not supported
            // in XML config. See SPR-12458 for details.
            //我们不能使用Profiles.of(...),因为XML配置中不支持配置文件表达式。有关详细信息,请参见SPR-12458。
            if (!getReaderContext().getEnvironment().acceptsProfiles(specifiedProfiles)) {
                if (logger.isDebugEnabled()) {
                    logger.debug("Skipped XML bean definition file due to specified profiles [" + profileSpec +
                                 "] not matching: " + getReaderContext().getResource());
                }
                return;
            }
        }
    }
    //之前做的事情
    preProcessXml(root);
    //开始解析
    parseBeanDefinitions(root, this.delegate);
    //完成beanDefinition之后做的事情
    postProcessXml(root);

    this.delegate = parent;
}
  • 创建一个delegate对象
  • 获取profile,进行校验
  • preProcessXml 设置解析之前做的事情
  • parseBeanDefinitions 开始i解析 (主要方法)
  • postProcessXml 设置解析之后做的事情

解析 org.springframework.beans.factory.xml.DefaultBeanDefinitionDocumentReader#parseBeanDefinitions

protected void parseBeanDefinitions(Element root, BeanDefinitionParserDelegate delegate) {
    //是不是根节点
    if (delegate.isDefaultNamespace(root)) {
        //获取子节点
        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;
                //是否是默认的命名空间 是不是 http://www.springframework.org/schema/beans
                if (delegate.isDefaultNamespace(ele)) {
                    //直接解析
                    parseDefaultElement(ele, delegate);
                }
                else {
                    //通过特殊处理进行解析
                    delegate.parseCustomElement(ele);
                }
            }
        }
    }
    else {
        //特殊解析
        delegate.parseCustomElement(root);
    }
}
  • 检查DOM的命名空间是不是http://www.springframework.org/schema/beans
    • 解析
  • 特殊解析

解析 org.springframework.beans.factory.xml.DefaultBeanDefinitionDocumentReader#parseDefaultElement

private void parseDefaultElement(Element ele, BeanDefinitionParserDelegate delegate) {
    //import
    if (delegate.nodeNameEquals(ele, IMPORT_ELEMENT)) {
        importBeanDefinitionResource(ele);
    }
    //alias
    else if (delegate.nodeNameEquals(ele, ALIAS_ELEMENT)) {
        processAliasRegistration(ele);
    }
    //bean
    else if (delegate.nodeNameEquals(ele, BEAN_ELEMENT)) {
        processBeanDefinition(ele, delegate);
    }
    //beans
    else if (delegate.nodeNameEquals(ele, NESTED_BEANS_ELEMENT)) {
        // recurse   
        doRegisterBeanDefinitions(ele);
    }
}
  • 解析import标签
  • 解析alias标签
  • 解析bean标签
  • 解析beans标签
    • 这里还需要调用org.springframework.beans.factory.xml.DefaultBeanDefinitionDocumentReader#doRegisterBeanDefinitions 递归解析

解析 org.springframework.beans.factory.xml.DefaultBeanDefinitionDocumentReader#processBeanDefinition

因为我们只看bean注册的过程,这里我们只看加载bean的

protected void processBeanDefinition(Element ele, BeanDefinitionParserDelegate delegate) {
    //获取BeanDefinition持有者  BeanDefinition的处理器
    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));
    }
}

我们先看org.springframework.beans.factory.xml.BeanDefinitionParserDelegate#parseBeanDefinitionElement(org.w3c.dom.Element) 方法

发现调用了自己的方法,

public BeanDefinitionHolder parseBeanDefinitionElement(Element ele) {
    return parseBeanDefinitionElement(ele, null);
}

下面来看ingframework.beans.factory.xml.BeanDefinitionParserDelegate#parseBeanDefinitionElement(org.w3c.dom.Element, org.springframework.beans.factory.config.BeanDefinition)

public BeanDefinitionHolder parseBeanDefinitionElement(Element ele, @Nullable BeanDefinition containingBean) {
    String id = ele.getAttribute(ID_ATTRIBUTE);
    String nameAttr = ele.getAttribute(NAME_ATTRIBUTE);

    //别名集合
    List<String> aliases = new ArrayList<>();
    if (StringUtils.hasLength(nameAttr)) {
        //有别名的集合话,就保存起来
        String[] nameArr = StringUtils.tokenizeToStringArray(nameAttr, MULTI_VALUE_ATTRIBUTE_DELIMITERS);
        aliases.addAll(Arrays.asList(nameArr));
    }

    String beanName = id;
    if (!StringUtils.hasText(beanName) && !aliases.isEmpty()) {
        beanName = aliases.remove(0);
        if (logger.isTraceEnabled()) {
            logger.trace("No XML 'id' specified - using '" + beanName +
                         "' as bean name and " + aliases + " as aliases");
        }
    }

    if (containingBean == null) {
        //验证beanname和别名是否已经被使用了
        checkNameUniqueness(beanName, aliases, ele);
    }

    //设置类信息
    AbstractBeanDefinition beanDefinition = parseBeanDefinitionElement(ele, beanName, containingBean);

    if (beanDefinition != null) {
        if (!StringUtils.hasText(beanName)) {
            try {
                //containingBean 不为空
                if (containingBean != null) {
                    //创建一个bean名称 在工厂中是唯一的
                    beanName = BeanDefinitionReaderUtils.generateBeanName(
                        beanDefinition, this.readerContext.getRegistry(), true);
                }
                else {//为空的话
                    //
                    beanName = this.readerContext.generateBeanName(beanDefinition);
                    //如果生成器返回类名加后缀,则注册普通bean类名的别名(如果可能的话)。
                    //这是Spring 1.2 / 2.0向后兼容性的预期。
                    String beanClassName = beanDefinition.getBeanClassName();
                    if (beanClassName != null &&
                        beanName.startsWith(beanClassName) && beanName.length() > beanClassName.length() &&
                        !this.readerContext.getRegistry().isBeanNameInUse(beanClassName)) {
                        aliases.add(beanClassName);
                    }
                }
                if (logger.isTraceEnabled()) {
                    logger.trace("Neither XML 'id' nor 'name' specified - " +
                                 "using generated bean name [" + beanName + "]");
                }
            }
            catch (Exception ex) {
                error(ex.getMessage(), ele);
                return null;
            }
        }
        //获取别名集合
        String[] aliasesArray = StringUtils.toStringArray(aliases);
        //创建一个当前BeanDefinition的处理器 包含了别名
        return new BeanDefinitionHolder(beanDefinition, beanName, aliasesArray);
    }

    return null;
}

接下来,我们来看 org.springframework.beans.factory.xml.BeanDefinitionParserDelegate#decorateBeanDefinitionIfRequired(org.w3c.dom.Element, org.springframework.beans.factory.config.BeanDefinitionHolder)

也是调用了内部的方法

public BeanDefinitionHolder decorateBeanDefinitionIfRequired(Element ele, BeanDefinitionHolder definitionHolder) {
    return decorateBeanDefinitionIfRequired(ele, definitionHolder, null);
}

org.springframework.beans.factory.xml.BeanDefinitionParserDelegate#decorateBeanDefinitionIfRequired(org.w3c.dom.Element, org.springframework.beans.factory.config.BeanDefinitionHolder, org.springframework.beans.factory.config.BeanDefinition)

public BeanDefinitionHolder decorateBeanDefinitionIfRequired(
			Element ele, BeanDefinitionHolder definitionHolder, @Nullable BeanDefinition containingBd) {

    BeanDefinitionHolder finalDefinition = definitionHolder;

    //首先根据自定义属性进行装饰。
    NamedNodeMap attributes = ele.getAttributes();
    for (int i = 0; i < attributes.getLength(); i++) {
        Node node = attributes.item(i);
        finalDefinition = decorateIfRequired(node, finalDefinition, containingBd);
    }

    //基于自定义嵌套元素进行装饰。
    NodeList children = ele.getChildNodes();
    for (int i = 0; i < children.getLength(); i++) {
        Node node = children.item(i);
        if (node.getNodeType() == Node.ELEMENT_NODE) {
            finalDefinition = decorateIfRequired(node, finalDefinition, containingBd);
        }
    }
    return finalDefinition;
}

一直到上面为止,加载步骤才完成,

下面的org.springframework.beans.factory.support.BeanDefinitionReaderUtils#registerBeanDefinition方法才是真正的注册

解析 org.springframework.beans.factory.support.BeanDefinitionReaderUtils#registerBeanDefinition

public static void registerBeanDefinition(
			BeanDefinitionHolder definitionHolder, BeanDefinitionRegistry registry)
			throws BeanDefinitionStoreException {

    //获取beanName
    String beanName = definitionHolder.getBeanName();
    //注册bean 到ioc中
    registry.registerBeanDefinition(beanName, definitionHolder.getBeanDefinition());

    //为bean名称注册别名(如果有)。
    String[] aliases = definitionHolder.getAliases();
    if (aliases != null) {
        for (String alias : aliases) {
            //注册别名
            registry.registerAlias(beanName, alias);
        }
    }
}
  • 获取beanName
  • 注册到ioc容器中
  • 如果有别名,别名注册到alias的容器中

我们来看注册到ioc容器的 方法org.springframework.beans.factory.support.DefaultListableBeanFactory#registerBeanDefinition

public void registerBeanDefinition(String beanName, BeanDefinition beanDefinition)
			throws BeanDefinitionStoreException {

    Assert.hasText(beanName, "Bean name must not be empty");
    Assert.notNull(beanDefinition, "BeanDefinition must not be null");

    //是否是AbstractBeanDefinition子类
    if (beanDefinition instanceof AbstractBeanDefinition) {
        try {
            //验证是否 是否已经有重写了方法,并且有工厂方法
            ((AbstractBeanDefinition) beanDefinition).validate();
        }
        catch (BeanDefinitionValidationException ex) {
            throw new BeanDefinitionStoreException(beanDefinition.getResourceDescription(), beanName,
                                                   "Validation of bean definition failed", ex);
        }
    }
    //根据beanName 从ioc中获取
    BeanDefinition existingDefinition = this.beanDefinitionMap.get(beanName);
    if (existingDefinition != null) {//存在的话
        if (!isAllowBeanDefinitionOverriding()) {//不允许覆盖就报错
            throw new BeanDefinitionOverrideException(beanName, beanDefinition, existingDefinition);
        }
        else if (existingDefinition.getRole() < beanDefinition.getRole()) {//存在的权限 小于创建的权限
            // e.g. was ROLE_APPLICATION, now overriding with ROLE_SUPPORT or ROLE_INFRASTRUCTURE
            if (logger.isInfoEnabled()) {
                logger.info("Overriding user-defined bean definition for bean '" + beanName +
                            "' with a framework-generated bean definition: replacing [" +
                            existingDefinition + "] with [" + beanDefinition + "]");
            }
        }
        else if (!beanDefinition.equals(existingDefinition)) { //2个是否相同
            if (logger.isDebugEnabled()) {
                logger.debug("Overriding bean definition for bean '" + beanName +
                             "' with a different definition: replacing [" + existingDefinition +
                             "] with [" + beanDefinition + "]");
            }
        }
        else {
            if (logger.isTraceEnabled()) {
                logger.trace("Overriding bean definition for bean '" + beanName +
                             "' with an equivalent definition: replacing [" + existingDefinition +
                             "] with [" + beanDefinition + "]");
            }
        }
        //最后进行保存
        this.beanDefinitionMap.put(beanName, beanDefinition);
    }
    else {
        if (hasBeanCreationStarted()) {//是否标记为创建创建过 从创建集合中 判断是否存在过
            //无法再修改启动时集合元素(用于稳定迭代)
            synchronized (this.beanDefinitionMap) {
                this.beanDefinitionMap.put(beanName, beanDefinition);//保存到IOC中
                //根据beanDefinitionNames集合创建啊一个修改后的Definitions集合
                List<String> updatedDefinitions = new ArrayList<>(this.beanDefinitionNames.size() + 1);
                //把原来的beanDefinitionNames 保存起来
                updatedDefinitions.addAll(this.beanDefinitionNames);
                //添加新加的beanName
                updatedDefinitions.add(beanName);
                //重新赋值
                this.beanDefinitionNames = updatedDefinitions;

                //查看是否已经存在单例对象的注册bean集合中了
                if (this.manualSingletonNames.contains(beanName)) {
                    //存在的话, 创建一个相同大小的Set
                    Set<String> updatedSingletons = new LinkedHashSet<>(this.manualSingletonNames);
                    //从中删除
                    updatedSingletons.remove(beanName);
                    //把修改后的集合重新赋值到manualSingletonNames
                    this.manualSingletonNames = updatedSingletons;
                }
            }
        }
        else { //其他情况
            //仍在启动注册阶段
            //保存
            this.beanDefinitionMap.put(beanName, beanDefinition);
            // 添加到ioc容器的names集合中
            this.beanDefinitionNames.add(beanName);
            //从单例对象的注册表中删除beanName
            this.manualSingletonNames.remove(beanName);
        }
        //清空
        this.frozenBeanDefinitionNames = null;
    }

    //存在过  或者 是单例的bean
    if (existingDefinition != null || containsSingleton(beanName)) {
        //重置
        //1 当前
        //2. 父名称
        resetBeanDefinition(beanName);
    }
}

总结

通过上面的代码。主要步骤在下面方法中

  • 定位

    • org.springframework.context.support.AbstractXmlApplicationContext
      • loadBeanDefinitions(org.springframework.beans.factory.support.DefaultListableBeanFactory) 通过XmlBeanDefinitionReader加载bean定义
      • loadBeanDefinitions(org.springframework.beans.factory.xml.XmlBeanDefinitionReader) 使用传入的XmlBeanDefinitionReader加载bean定义。
    • org.springframework.beans.factory.support.AbstractBeanDefinitionReader
      • loadBeanDefinitions(org.springframework.core.io.Resource…) 从传入的资源中加载bean定义
    • org.springframework.beans.factory.xml.XmlBeanDefinitionReader
      • loadBeanDefinitions(org.springframework.core.io.Resource) 从传入的资源中加载bean定义 (xml)
      • loadBeanDefinitions(org.springframework.core.io.support.EncodedResource) 从传入的资源中加载bean定义 (xml)
      • doLoadBeanDefinitions 开始执行 从指定的XML文件加载bean定义。
      • registerBeanDefinitions 注册DOM文档中包含的bean定义。
  • 加载

    • org.springframework.beans.factory.xml.DefaultBeanDefinitionDocumentReader
      • registerBeanDefinitions 根据“spring-beans”XSD(或历史上的DTD)解析bean定义。
      • parseBeanDefinitions 解析文档中根级别的元素:“import”,“alias”,“bean”。
      • parseDefaultElement 根据传入的标签进行解析
      • processBeanDefinition 处理传入的bean元素,解析bean定义并将其注册到注册表。
      • registerBeanDefinition 注册最终的BeanDefinition
  • 注册

    • org.springframework.beans.factory.support.DefaultListableBeanFactory
      • registerBeanDefinition 完成注册
  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值