`
wang4674890
  • 浏览: 88956 次
  • 性别: Icon_minigender_2
  • 来自: 厦门
社区版块
存档分类
最新评论

spring源代码分析(2)--BeanFactory

 
阅读更多

我们首先来看下BeanFacroty接口

 

  1. package org.springframework.beans.factory;
  2. import org.springframework.beans.BeansException;
  3. public interface BeanFactory {
  4.     
  5.     String FACTORY_BEAN_PREFIX = "&";
  6.     Object getBean(String name) throws BeansException;
  7.     Object getBean(String name, Class requiredType) throws BeansException;
  8.     
  9.     boolean containsBean(String name);
  10.     String[] getAliases(String name) throws NoSuchBeanDefinitionException;
  11. }


可以看出,BeanFactory定义了Factory的基本方法,他能够生成Bean以及辨别Bean是否包含在factory中,以及从一个Bean的名字的到其别名;

  1. package org.springframework.beans.factory;
  2. public interface HierarchicalBeanFactory extends BeanFactory {
  3.     
  4.     BeanFactory getParentBeanFactory();
  5.     boolean containsLocalBean(String name);
  6. }

HierarchicalBeanFactory在beanFactory的基础上,提供了BeanFactory能够集成父容器的功能,从而形成了工厂链;

  1. public interface ConfigurableBeanFactory extends HierarchicalBeanFactory {
  2.     void setParentBeanFactory(BeanFactory parentBeanFactory);
  3.     
  4.     void registerCustomEditor(Class requiredType, PropertyEditor propertyEditor);
  5.     
  6.     void addBeanPostProcessor(BeanPostProcessor beanPostProcessor);
  7.     
  8.     int getBeanPostProcessorCount();
  9.     
  10.     void registerAlias(String beanName, String alias) throws BeansException;
  11.     
  12.     void registerSingleton(String beanName, Object singletonObject) throws BeansException;
  13.     
  14.     boolean containsSingleton(String beanName);
  15.     
  16.     void destroySingletons();
  17. }

ConfigurableBeanFactory有继承了HierarchicalBeanFactory 接口,在HierarchicalBeanFactory 接口的基础上,有实现了BeanFactory的一些可配置功能,比如说增加VBeanPostProcessor,注册别名,注册单列等等;

  1. public interface ListableBeanFactory extends BeanFactory {
  2.     
  3.     boolean containsBeanDefinition(String beanName);
  4.     
  5.     int getBeanDefinitionCount();
  6.     
  7.     String[] getBeanDefinitionNames();
  8.     
  9.     
  10.     Map getBeansOfType(Class type, boolean includePrototypes, boolean includeFactoryBeans)
  11.         throws BeansException;
  12. }

而ListableBeanFactory能够列出此工厂中的BeanDifinition的信息;一个BeanDifinition就是一个Bean,他不仅包含了Bean对象,并且他里面每一个配置文件就是其的一个属性,比如lazy-init,dependcy-check等等;

在这里,我们可以先来看看BeanDifinition;

可见,在接口设计的时候,先抽象出来接口,由于子类之间有共同的功能实现,那么,再次抽象出抽象类,尽量把相同的代码往上推,其次,最终实现;

ChildBeanDefinition和RootBeanDefinition分别表示两种BeanDefition类型,ChildBeanDefinition表示继承了别的Bean的配置信息的Bean元素,所以,他有一个直接的
    private final String parentName;

       public String getParentName() {
        return parentName;
    }

所以,在得到这个bean的时候,我们必须一层一层的往上组合merge才能得到完整的Bean的配置信息;

而大多数时候,我们使用RootBeanFactory,表示,这个Bean所具有的配置信息完全由他自己所定义;


弄清楚这个以后,我们从XmlBeanFactory来反推查看类结构;

在这里,我们可以看出类图结构错综复杂,很是繁琐,有些类在借口的实现过程中间出现了重复,我们来从上而下的分析这类图:

(1)AutowireCapableBeanFactory:实现了自动装配的功能,在此接口中,定义了几种自动装配的类型,比如:

  1.     int AUTOWIRE_BY_NAME = 1;
  2.     int AUTOWIRE_BY_TYPE = 2;
  3.     
  4.     int AUTOWIRE_CONSTRUCTOR = 3;
  5.     int AUTOWIRE_AUTODETECT = 4;

并且能够检查Bean自动装配的BeanPostProcessors的postProcessAfter/BeforeInitialization并且应用;

  1.     Object applyBeanPostProcessorsBeforeInitialization(Object existingBean, String beanName)
  2.             throws BeansException;
  3.     Object applyBeanPostProcessorsAfterInitialization(Object existingBean, String beanName)
  4.             throws BeansException;

AbstractBeanFactory则是上述接口的 一个抽象实现,很多方法都在此方法中得到了实现;

首先是BeanFactory功能的实现;如:

  1. public Object getBean(String name, Class requiredType, Object[] args) throws BeansException {
  2.         String beanName = transformedBeanName(name);
  3.      //对名字进行了预处理,去掉了“&”
  4.         Object bean = null;
  5.         // Eagerly check singleton cache for manually registered singletons.
  6.         Object sharedInstance = null;
  7.         synchronized (this.singletonCache) {
  8.         //是否是共享的实例;若是,则从缓存中取出;
  9.             sharedInstance = this.singletonCache.get(beanName);
  10.         }
  11.         if (sharedInstance != null) {
  12.            //此例是否正在生成?
  13.             if (isSingletonCurrentlyInCreation(beanName)) {
  14.                 if (logger.isDebugEnabled()) {
  15.                     logger.debug("Returning eagerly cached instance of singleton bean '" + beanName +
  16.                             "' that is not fully initialized yet - a consequence of a circular reference");
  17.                 }
  18.             }
  19.             else {
  20.                 if (logger.isDebugEnabled()) {
  21.                     logger.debug("Returning cached instance of singleton bean '" + beanName + "'");
  22.                 }
  23.             }
  24.             bean = getObjectForSharedInstance(name, sharedInstance);
  25.         }
  26.         else {
  27.             // Fail if we're already creating this singleton instance:
  28.             // We're assumably within a circular reference.
  29.             if (isSingletonCurrentlyInCreation(beanName)) {
  30.                 throw new BeanCurrentlyInCreationException(beanName);
  31.             }
  32.             // Check if bean definition exists in this factory.
  33.             if (getParentBeanFactory() != null && !containsBeanDefinition(beanName)) {
  34.                 // Not found -> check parent.
  35.                 if (getParentBeanFactory() instanceof AbstractBeanFactory) {
  36.                     // Delegation to parent with args only possible for AbstractBeanFactory.
  37.                     return ((AbstractBeanFactory) getParentBeanFactory()).getBean(name, requiredType, args);
  38.                 }
  39.                 else if (args == null) {
  40.                     // No args -> delegate to standard getBean method.
  41.                     //通过父容器来取得Bean
  42.                     return getParentBeanFactory().getBean(name, requiredType);
  43.                 }
  44.                 else {
  45.                     throw new NoSuchBeanDefinitionException(beanName,
  46.                             "Cannot delegate to parent BeanFactory because it does not supported passed-in arguments");
  47.                 }
  48.             }
  49.             //得到一个完整的BeanDefinition;
  50.             RootBeanDefinition mergedBeanDefinition = getMergedBeanDefinition(beanName, false);
  51.             checkMergedBeanDefinition(mergedBeanDefinition, beanName, requiredType, args);
  52.             // Create bean instance.
  53.             //如果是单例,那么就添加在单例map中去;
  54.             //在这里,你可以看成享元模式的一种灵活应用,
  55.             if (mergedBeanDefinition.isSingleton()) {
  56.                 synchronized (this.singletonCache) {
  57.                     // Re-check singleton cache within synchronized block.
  58.                     sharedInstance = this.singletonCache.get(beanName);
  59.                     if (sharedInstance == null) {
  60.                         if (logger.isDebugEnabled()) {
  61.                             logger.debug("Creating shared instance of singleton bean '" + beanName + "'");
  62.                         }
  63.                         //标示一下,此实例Bean正在生成,以免Sington重复创建,变相达到线程安全;
  64.                         this.currentlyInCreation.add(beanName);
  65.                         try {
  66.                             sharedInstance = createBean(beanName, mergedBeanDefinition, args);
  67.                             addSingleton(beanName, sharedInstance);
  68.                         }
  69.                         catch (BeansException ex) {
  70.                             // Explicitly remove instance from singleton cache: It might have been put there
  71.                             // eagerly by the creation process, to allow for circular reference resolution.
  72.                             // Also remove any beans that received a temporary reference to the bean.
  73.                             destroyDisposableBean(beanName);
  74.                             throw ex;
  75.                         }
  76.                         finally {
  77.                             //beanDefinition生成完毕,从正在创建数组中移除
  78.                             this.currentlyInCreation.remove(beanName);
  79.                         }
  80.                     }
  81.                 }
  82.                 bean = getObjectForSharedInstance(name, sharedInstance);
  83.             }
  84.             else {
  85.                 //如果是原型Bean,,那么我们直接应用这个类型来生成一个实例
  86.                 bean = createBean(beanName, mergedBeanDefinition, args);
  87.             }
  88.         }
  89.         // Check if required type matches the type of the actual bean instance.
  90.         if (requiredType != null && !requiredType.isAssignableFrom(bean.getClass())) {
  91.             throw new BeanNotOfRequiredTypeException(name, requiredType, bean.getClass());
  92.         }
  93.         return bean;
  94.     }


(2)HierarchicalBeanFactory的实现:

  1. public BeanFactory getParentBeanFactory() {
  2.         return parentBeanFactory;
  3.     }
  4.     public boolean containsLocalBean(String name) {
  5.         String beanName = transformedBeanName(name);
  6.         return (containsSingleton(beanName) || containsBeanDefinition(beanName));
  7.     }

判断一个Bean是否在本地配置;而不是在父工厂中生成;

我们可以抽取出mergeBeanDefinition来看看:

  1.     protected RootBeanDefinition getMergedBeanDefinition(String beanName, BeanDefinition bd)
  2.             throws BeanDefinitionStoreException {
  3.         //如果是根Bean.则返回,退出嵌套;
  4.         if (bd instanceof RootBeanDefinition) {
  5.             // Return root bean definition as-is.
  6.             return (RootBeanDefinition) bd;
  7.         }
  8.         else if (bd instanceof ChildBeanDefinition) {
  9.             // Child bean definition: needs to be merged with parent.
  10.             ChildBeanDefinition cbd = (ChildBeanDefinition) bd;
  11.             RootBeanDefinition pbd = null;
  12.             try {
  13.                 if (!beanName.equals(cbd.getParentName())) {
  14.                 //在这里进行了嵌套;
  15.                     pbd = getMergedBeanDefinition(cbd.getParentName(), true);
  16.                 }
  17.                 else {
  18.                     if (getParentBeanFactory() instanceof AbstractBeanFactory) {
  19.                         AbstractBeanFactory parentFactory = (AbstractBeanFactory) getParentBeanFactory();
  20.                         pbd = parentFactory.getMergedBeanDefinition(cbd.getParentName(), true);
  21.                     }
  22.                     else {
  23.                         throw new NoSuchBeanDefinitionException(cbd.getParentName(),
  24.                                 "Parent name '" + cbd.getParentName() + "' is equal to bean name '" + beanName +
  25.                                 "': cannot be resolved without an AbstractBeanFactory parent");
  26.                     }
  27.                 }
  28.             }
  29.             catch (NoSuchBeanDefinitionException ex) {
  30.                 throw new BeanDefinitionStoreException(cbd.getResourceDescription(), beanName,
  31.                         "Could not resolve parent bean definition '" + cbd.getParentName() + "'", ex);
  32.             }
  33.             // Deep copy with overridden values.
  34.             RootBeanDefinition rbd = new RootBeanDefinition(pbd);
  35.             rbd.overrideFrom(cbd);
  36.             // Validate merged definition: mainly to prepare method overrides.
  37.             try {
  38.                 rbd.validate();
  39.             }
  40.             catch (BeanDefinitionValidationException ex) {
  41.                 throw new BeanDefinitionStoreException(rbd.getResourceDescription(), beanName,
  42.                         "Validation of bean definition failed", ex);
  43.             }
  44.             return rbd;
  45.         }
  46.         else {
  47.             throw new BeanDefinitionStoreException(bd.getResourceDescription(), beanName,
  48.                     "Definition is neither a RootBeanDefinition nor a ChildBeanDefinition");
  49.         }
  50.     }
  • sharedInstance = createBean(beanName, mergedBeanDefinition, args);
  •                             addSingleton(beanName, sharedInstance);

 


这段代码大量的应用了模板模式:createBean是在AbstractAutowireCapableBeanFactory类中实现的:
protected Object createBean(String beanName, RootBeanDefinition mergedBeanDefinition, Object[] args)
            throws BeanCreationException {

        // Guarantee initialization of beans that the current one depends on.
        if (mergedBeanDefinition.getDependsOn() != null) {
            for (int i = 0; i < mergedBeanDefinition.getDependsOn().length; i++) {
             //将依赖的类加载进来;
                getBean(mergedBeanDefinition.getDependsOn()[i]);
            }
        }


bean = applyBeanPostProcessorsBeforeInstantiation(mergedBeanDefinition.getBeanClass(), beanName);
会为这个Bean生成一个代理,每次都会调用拦截的方法;
AbstractAutoProxyCreator:

public Object postProcessBeforeInstantiation(Class beanClass, String beanName) throws BeansException {
        if (isInfrastructureClass(beanClass, beanName)) {
            return null;
        }

        // Create proxy here if we have a custom TargetSource.
        // Suppresses unnecessary default instantiation of the target bean:
        // The TargetSource will handle target instances in a custom fashion.
        TargetSource targetSource = getCustomTargetSource(beanClass, beanName);
        if (targetSource != null) {
            Object[] specificInterceptors = getAdvicesAndAdvisorsForBean(beanClass, beanName, targetSource);
            //生成代理
            return createProxy(beanClass, beanName, specificInterceptors, targetSource);
        }

        return null;
    }


在createBean的代码中,我们可以看见这么一段:

            populateBean(beanName, mergedBeanDefinition, instanceWrapper);

            if (bean instanceof BeanNameAware) {
                if (logger.isDebugEnabled()) {
                    logger.debug("Invoking setBeanName on BeanNameAware bean '" + beanName + "'");
                }
                ((BeanNameAware) bean).setBeanName(beanName);
            }

            if (bean instanceof BeanFactoryAware) {
                if (logger.isDebugEnabled()) {
                    logger.debug("Invoking setBeanFactory on BeanFactoryAware bean '" + beanName + "'");
                }
                ((BeanFactoryAware) bean).setBeanFactory(this);
            }

            originalBean = bean;
            bean = applyBeanPostProcessorsBeforeInitialization(bean, beanName);
            invokeInitMethods(beanName, bean, mergedBeanDefinition);
            bean = applyBeanPostProcessorsAfterInitialization(bean, beanName);

在这个的代码中,我们可以清晰的看见Bean被实例化所要经过的一些过程;其中包括处理回调接口,调用init方法以及拦截器等等;

其次:
在最终的XmlBeanDefinition中:

public XmlBeanFactory(Resource resource, BeanFactory parentBeanFactory) throws BeansException {
        super(parentBeanFactory);
        this.reader.loadBeanDefinitions(resource);
    }

这个Reader.load()实际上是一个XML解析器,他的功能是把xml配置文件转换成为BeanDefinition存储到BeanFactory的
BeanDefinition map中去;

BeanFactory的类图结构,大量的采用模板模式,在顶级的抽象类中间约定了算法的实际流程,而把算法的具体实现方式等等推到了子类实现,从而可以扩展出多个功能的工厂

 

 

分享到:
评论

相关推荐

    Spring源代码解析

    Spring框架是Java开发中的...总结来说,Spring源代码解析涵盖了IoC容器的运作、AOP的实现、数据访问的策略以及Web MVC的流程。深入研究这些内容,将有助于Java开发者成为Spring框架的专家,并在实际项目中游刃有余。

    Spring源代码解析(一):IOC容器.doc

    在Spring源代码解析的第一部分,我们将聚焦于IOC容器,特别是BeanFactory接口,它是所有Spring容器的基础。 BeanFactory接口是Spring的基石,它定义了基本的容器操作,如获取Bean、检查Bean是否存在、确定Bean的...

    spring源码分析(1-10)

    Spring 源代码分析系列涵盖了多个关键模块,包括事务处理、IoC容器、JDBC、MVC、AOP以及与Hibernate和Acegi安全框架的集成。以下是对这些知识点的详细阐述: 1. **Spring 事务处理**:Spring 提供了声明式事务管理...

    spring官方完整jar包-4.1.6版

    3. **面向切面编程(AOP)**:Spring的AOP模块支持在不修改源代码的情况下,对应用程序进行横切关注点(如日志、事务管理)的插入。这有助于保持代码的整洁,提高可复用性。 4. **Bean工厂与ApplicationContext**:...

    简单Spring框架模拟器--Demo

    接着,“tiny-spring-step-2-abstract-beanfactory-and-do-bean-initilizing-in-it.zip”进一步深入,介绍了AbstractBeanFactory和Bean的初始化过程。在这个阶段,你将学习到BeanFactory如何解析XML配置,创建Bean...

    spring源代码解析

    代码解析2,部分摘抄 简单的说,在web容器中,通过ServletContext为Spring的IOC容器提供宿主环境,对应的建立起一个IOC容器的体系。其中,首先需要建立的是根上下文,这个上下文持有的对象可以有业务对象,数据存取...

    spring源代码下载地址

    首先,Spring源代码的下载对于深入理解其工作原理、自定义扩展或排查问题至关重要。在描述中提到的"Spring源代码的下载地址",通常是指Spring官方仓库或者第三方镜像站点。Spring的源代码托管在GitHub上,你可以访问...

    spring源代码下载(共3部分)

    本篇将详细讲解Spring源代码下载的重要性和如何深入学习Spring的源代码。 首先,了解Spring源代码能够帮助开发者深入理解框架的工作原理,这对于优化性能、解决复杂问题以及自定义扩展都有极大的帮助。Spring框架由...

    spring源代码

    源代码分析对于深入理解其工作原理、优化应用性能以及进行二次开发至关重要。让我们一起深入探究Spring源码中的关键知识点。 首先,Spring的核心组件之一是IoC容器。IoC容器通过反转控制权,使得对象的创建和依赖...

    spring-framework-5.3.8

    Spring 提供了基于代理的 AOP 实现,可以在不修改源代码的情况下增强已有类的功能。 4. **数据访问**:Spring 支持多种数据访问技术,如 JDBC、JPA、Hibernate 和 MyBatis。在 5.3.8 版本中,可能对这些数据访问...

    Spring3.0源代码

    源代码分析通常涉及以下几个关键知识点: 1. **依赖注入**:Spring3.0的核心特性之一,允许开发者通过配置来管理对象之间的依赖关系,而不是硬编码这些依赖。在`src`目录下,可以找到`org.springframework.beans`和...

    spring-framework-3.0.5-dist.rar

    2. **AOP模块**:Spring的AOP支持面向切面的编程,允许开发者定义方法拦截器和切面,以实现非侵入式的服务,如事务管理、日志记录等。 3. **数据访问/集成**:Spring提供了对JDBC、ORM(Object-Relational Mapping...

    spring-framework-src-2.5.6.rar

    2. **org.springframework.beans**:处理Bean工厂(BeanFactory)和Bean的生命周期管理,包括属性注入、类型转换等。 3. **org.springframework.aop**:实现了AOP的概念,允许开发者定义切面并将其应用到目标对象上...

    Spring框架核心源代码的分析及其感受-6

    在源代码中,`org.springframework.beans.factory.BeanFactory` 和 `org.springframework.context.ApplicationContext` 是IoC容器的两个主要接口,它们提供了加载配置、获取Bean和处理依赖注入等功能。通过阅读这些...

    spring-developing-java-applications-enterprise

    - 创建源代码文件,并在其中定义Spring管理的bean。 - 创建Spring的bean配置文件,定义bean之间的依赖关系。 5. **运行程序**:通过配置好的Spring容器来启动程序,Spring会自动管理和初始化bean。 #### 六、控制...

    spring-framework-3.0.5.RELEASE-src

    开发者可以通过阅读源代码,了解如何通过BeanFactory和ApplicationContext实现依赖的自动装配,以及如何使用@Autowired、@Qualifier等注解来精确控制依赖的注入。 其次,Spring AOP(面向切面编程)是其另一大亮点...

    spring 源代码解析.zip

    《Spring源代码解析》 Spring框架作为Java领域最流行的开源框架之一,它的设计思想和实现方式一直是广大开发者关注的焦点。深入理解Spring的源代码,能够帮助我们更好地掌握其工作原理,提高我们的开发效率和代码...

    达内java培训SPRING 源代码

    在这个"达内java培训SPRING 源代码"中,初学者可以深入理解Spring框架的基础和核心概念。 1. **依赖注入**:在Spring中,依赖注入是一种设计模式,它允许对象之间的依赖关系在运行时被管理,而不是在代码中硬编码。...

    好多人找不到源代码,此处提供spring源代码

    Spring源代码的深入理解对于提升开发技能、优化代码以及解决问题具有重要意义。在这个压缩包中,包含的是Spring框架的核心模块——`org`目录,这通常是Spring框架的主要组织结构。 在`org`目录下,我们可以找到...

    《精通Spring2.x-企业应用开发详解》源代码

    源代码涵盖了第二至第六章的内容,这五个章节分别涉及了Spring的基础、依赖注入、AOP(面向切面编程)、事务管理和Web应用等方面的知识。以下是这些章节的关键知识点详解: 1. **第二章:Spring概述** - Spring...

Global site tag (gtag.js) - Google Analytics