原创内容,转载请注明出处
1、BeanFactory类图
BeanFactory类图如下
从上图可以看出BeanFactory主要实现类是XmlBeanFactory(Spring3.1建议弃用,可以使用DefaultListableBeanFactory和XmlBeanDefinitionReader编程实现)和DefaultListableBeanFactory。
2、BeanFactory编程式实现
基本IOC容器BeanFactory的编程式实现方式大致可分为三个步骤
1.加载xml资源 Resource定位xml
2.BeanDefinition读取和加载 BeanDefinitionParserDelegate代理对象去读取
3.BeanDefinition注册 BeanDefinitionRegistry去注册
创建bean.xml,内容如下
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://www.springframework.org/schema/beans" xmlns:aop="http://www.springframework.org/schema/aop" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.0.xsd"> <bean id="person" class="com.test.bean.Person"> <property name="name" value="ylxy"/> <property name="age" value="25"/> </bean> </beans>
创建Person类,代码如下
package com.test.bean; public class Person { private String name; private int age; public String getName() { return name; } public void setName(String name) { this.name = name; } public int getAge() { return age; } public void setAge(int age) { this.age = age; } public void info(){ System.out.println("name:"+getName()+" age:"+getAge()); } }
创建junit测试代码,内容如下
@Test public void testBeanFactory(){ ClassPathResource resource = new ClassPathResource("bean.xml"); DefaultListableBeanFactory bf = new DefaultListableBeanFactory(); XmlBeanDefinitionReader reader = new XmlBeanDefinitionReader(bf); reader.loadBeanDefinitions(resource); Person p = bf.getBean("person", Person.class); p.info(); }
最终测试结果如下
3、IOC配置文件处理(BeanDefinition解析与注册)
从编程代码上看,可分以下几个步骤
1.Resource定位(即Resource包装spring xml配置文件)。
2.BeanDefinition解析和注册(由XmlBeanDefinitionReader类处理解析和注册)。
3.获取Bean(由BeanFactory获取,依赖注入)。
从上面可知BeanDefinition的解析和注册入口在XmlBeanDefinitionReader的loadBeanDefinitions(Resource resource)方法上。
1.在loadBeanDefinitions方法中,调用该类的doLoadBeanDefinitions(InputSource inputSource, Resource resource)方法。
2.在doLoadBeanDefinitions方法中通过DocumentLoader(实际是DefaultDocumentLoader对象)加载xml文件成Document类,之后调用该类的registerBeanDefinitions(Document doc, Resource resource)方法去解析加载和注册BeanDefinition。
3.在registerBeanDefinitions方法中创建DefaultBeanDefinitionDocumentReader对象,并调用registerBeanDefinitions( Document doc, XmlReaderContext readerContext)去解析注册BeanDefinition,解析时会对不同的xml元素进行不同的解析,比如<import/><alias/><bean/><beans/>等元素。
4.最终实际解析BeanDefinition的实现是创建BeanDefinitionParserDelegate委托对象,调用方法parseBeanDefinitionElement(Element ele)去解析BeanDefinition。
5.注册对象是在BeanDefinitionReaderUtils.registerBeanDefinition( BeanDefinitionHolder definitionHolder, BeanDefinitionRegistry registry)中实现。
BeanDefinition解析实现:调用BeanDefinitionParserDelegate.parseBeanDefinitionElement(Element ele)去解析xml成BeanDefinition,并将BeanDefinition封装成BeanDefinitionHolder。
BeanDefinition注册实现:调用BeanDefinitionRegistry.registerBeanDefinition(String beanName, BeanDefinition beanDefinition),由于DefaultListableBeanFactory实现了该接口的方法,故而实现注册BeanDefinition是在DefaultListableBeanFactory中。在该类中通过将BeanDefinition存储到一个HashMap集合中,完成对BeanDefinition的注册。
4、依赖注入
依赖注入和控制反转是同一个概念。当某个A javaBean需要B javaBean的引用去协助处理一些事情的时候,在传统程序设计中,是需要A去new一个B对象。然而在spring中却不一样,创建B对象的工作不是由A来new,而是由spring IOC容器去创建,然后将创建的B对象注入到A中。
spring的依赖注入入口也就是BeanFactory的getBean()方法。getBean()方法最后调用doGetBean()方法,即doGetBean()方法是依赖注入的真实入口,代码如下。
protected <T> T doGetBean( final String name, final Class<T> requiredType, final Object[] args, boolean typeCheckOnly) throws BeansException { final String beanName = transformedBeanName(name); Object bean; // Eagerly check singleton cache for manually registered singletons. Object sharedInstance = getSingleton(beanName); if (sharedInstance != null && args == null) { if (logger.isDebugEnabled()) { if (isSingletonCurrentlyInCreation(beanName)) { logger.debug("Returning eagerly cached instance of singleton bean '" + beanName + "' that is not fully initialized yet - a consequence of a circular reference"); } else { logger.debug("Returning cached instance of singleton bean '" + beanName + "'"); } } bean = getObjectForBeanInstance(sharedInstance, name, beanName, null); } else { // Fail if we're already creating this bean instance: // We're assumably within a circular reference. if (isPrototypeCurrentlyInCreation(beanName)) { throw new BeanCurrentlyInCreationException(beanName); } // Check if bean definition exists in this factory. BeanFactory parentBeanFactory = getParentBeanFactory(); //检查这个bean是否在当前这个BeanFactory中,如果当前BeanFactory的不包含这个bean,则去父级BeanFactory查找。 if (parentBeanFactory != null && !containsBeanDefinition(beanName)) { // Not found -> check parent. String nameToLookup = originalBeanName(name); if (args != null) { // Delegation to parent with explicit args. return (T) parentBeanFactory.getBean(nameToLookup, args); } else { // No args -> delegate to standard getBean method. return parentBeanFactory.getBean(nameToLookup, requiredType); } } if (!typeCheckOnly) { markBeanAsCreated(beanName); } try { final RootBeanDefinition mbd = getMergedLocalBeanDefinition(beanName); checkMergedBeanDefinition(mbd, beanName, args); // Guarantee initialization of beans that the current bean depends on. String[] dependsOn = mbd.getDependsOn(); //查找这个BeanDefinition是否有依赖对象,如果有依赖的对象,则IOC容器先创建依赖对象。 if (dependsOn != null) { for (String dependsOnBean : dependsOn) { getBean(dependsOnBean); registerDependentBean(dependsOnBean, beanName); } } // Create bean instance. if (mbd.isSingleton()) { //获取单例对象 sharedInstance = getSingleton(beanName, new ObjectFactory<Object>() { public Object getObject() throws BeansException { try { //创建Bean的入口方法 return createBean(beanName, mbd, args); } catch (BeansException ex) { // Explicitly remove instance from singleton cache: It might have been put there // eagerly by the creation process, to allow for circular reference resolution. // Also remove any beans that received a temporary reference to the bean. destroySingleton(beanName); throw ex; } } }); //这里创建动态代理对象 bean = getObjectForBeanInstance(sharedInstance, name, beanName, mbd); } else if (mbd.isPrototype()) { // It's a prototype -> create a new instance. Object prototypeInstance = null; try { beforePrototypeCreation(beanName); prototypeInstance = createBean(beanName, mbd, args); } finally { afterPrototypeCreation(beanName); } bean = getObjectForBeanInstance(prototypeInstance, name, beanName, mbd); } else { String scopeName = mbd.getScope(); final Scope scope = this.scopes.get(scopeName); if (scope == null) { throw new IllegalStateException("No Scope registered for scope '" + scopeName + "'"); } try { Object scopedInstance = scope.get(beanName, new ObjectFactory<Object>() { public Object getObject() throws BeansException { beforePrototypeCreation(beanName); try { return createBean(beanName, mbd, args); } finally { afterPrototypeCreation(beanName); } } }); bean = getObjectForBeanInstance(scopedInstance, name, beanName, mbd); } catch (IllegalStateException ex) { throw new BeanCreationException(beanName, "Scope '" + scopeName + "' is not active for the current thread; " + "consider defining a scoped proxy for this bean if you intend to refer to it from a singleton", ex); } } } catch (BeansException ex) { cleanupAfterBeanCreationFailure(beanName); throw ex; } } // Check if required type matches the type of the actual bean instance. if (requiredType != null && bean != null && !requiredType.isAssignableFrom(bean.getClass())) { try { return getTypeConverter().convertIfNecessary(bean, requiredType); } catch (TypeMismatchException ex) { if (logger.isDebugEnabled()) { logger.debug("Failed to convert bean '" + name + "' to required type [" + ClassUtils.getQualifiedName(requiredType) + "]", ex); } throw new BeanNotOfRequiredTypeException(name, requiredType, bean.getClass()); } } return (T) bean; }
从上述代码可以看出创建对象是由DefaultSingletonBeanRegistry的getSingleton方法去创建,然后回调ObjectFactory的getObject方法,ObjectFactory的getObject方法又调用了createBean方法。也就是说实际创建对象的方法是AbstractAutowireCapableBeanFactory类的createBean方法,该方法内又调用doCreateBean去创建Bean。
1.调用createBeanInstance方法去创建BeanWrap(BeanWrap将Bean实例包装了),该方法通过Java反射机制调用Bean的构造函数,完成Bean对象的创建,并将Bean包装到一个BeanWrap对象中。
2.调用populateBean方法,给bean的属性初始化,也就是<bean/>标签中配置的属性值。
3.调用initializeBean方法,初始化Bean。
3.1.invokeAwareMethods方法,如果Bean实现了Aware,则调用对应的Aware接口方法。
3.2.applyBeanPostProcessorsBeforeInitialization方法,如果BeanFactory配置了BeanPostProcessor,则执行BeanPostProcessor的postProcessBeforeInitialization方法。
3.3.invokeInitMethods方法,如果Bean实现了InitializingBean接口,则调用执行InitializingBean的接口方法。
3.4.applyBeanPostProcessorsAfterInitialization方法,如果BeanFactory配置了BeanPostProcessor,则执行BeanPostProcessor的postProcessAfterInitialization方法。
4.registerDisposableBeanIfNecessary方法,如果Bean实现了DisposableBean接口或者Bean需要destroy方法,则给Bean注册DisposableBean功能(即把bean名称和销毁方法保存在一个Map集合中)。
5.getObjectForBeanInstance(sharedInstance, name, beanName, mbd)方法,创建bean完成后,如果创建的bean实例是一个FactoryBean(相当于工厂模式的Factory,用来生产Bean实例),则会再调用FactoryBean的getObject()方法来生成实际需要的bean实例。也就是说如果getBean("factoryBean")会首先获取FactoryBean实例,并且在通过FactoryBean工厂的getObject方法去获取Bean,如果是getBean(“&factoryBean”),则spring会直接返回一个FactoryBean实例。
5、总结
对于BeanDefinition的解析和注册过程,首先将xml配置文件加载进Resource中,然后将Resource解析成Document对象,最后通过BeanDefinitionParserDelegate委托对象去解析Document对象,并将解析的BeanDefinition包装成BeanDefinitionHolder。最后通过BeanDefinitionRegistry接口将BeanDefinition存储到HashMap集合中,以完成BeanDefinition的的注册,具体实现在DefaultListableBeanFactory类中,该类实现了BeanDefinitionRegistry接口。
对于依赖注入,spring主要是使用Java的反射机制来创建bean对象,并为bean进行一些初始化操作。
源代码如附件
相关推荐
《Spring源码分析——ApplicationContext》 在Java世界中,Spring框架是不可或缺的一部分,它以其强大的IoC(Inversion of Control,控制反转)和AOP(Aspect Oriented Programming,面向切面编程)特性,极大地...
### Spring 源码分析——设计模式篇 #### 一、引言 Spring框架作为Java企业级开发领域中不可或缺的一部分,其内部集成了多种设计模式,不仅有助于提高系统的可维护性和扩展性,还能够帮助开发者更好地理解和应用...
### Spring源码分析_Spring_IOC:深入理解Spring的IOC容器机制 #### 基本概念与核心作用 在探讨Spring框架的核心组件之一——IOC(Inversion of Control,控制反转)容器之前,首先需要理解它在Spring框架中的角色...
在源码分析的过程中,读者会深入理解Spring的内部工作机制,例如如何解析配置、如何创建bean实例、如何实现AOP代理等。这将有助于开发者编写更高效、更健壮的代码,也能为参与Spring的扩展或定制打下坚实基础。 总...
本压缩包“Spring源码解析”提供了对Spring框架核心组件——IOC(Inversion of Control,控制反转)、AOP(Aspect Oriented Programming,面向切面编程)以及Transaction(事务管理)的源码分析,帮助开发者更全面地...
《Spring框架源码分析——基于UML图的解读》 在深入探讨Spring框架源码之前,我们首先要理解什么是UML(统一建模语言)。UML是一种标准的图形化建模语言,用于软件设计和系统分析,它通过图表来表示系统的结构、...
### Spring源码分析之IOC容器深入探讨 #### 一、Spring框架及IOC容器概述 Spring框架作为企业级Java开发中最常用的轻量级框架之一,以其强大的功能和灵活的设计深受开发者们的喜爱。Spring的核心特性之一便是...
在深入理解Spring框架的过程中,源码分析是不可或缺的一环。本文将对Spring的IOC(Inversion of Control,控制反转)容器的源码进行初步探讨,旨在帮助读者从整体上把握其设计理念和实现方式。 Spring的核心就是IOC...
总之,通过深入学习Spring源码,开发者不仅可以提升编程能力,还能掌握更多软件设计和架构方面的知识,对于职业发展大有裨益。在实际开发中,结合源码理解,能够更好地解决遇到的问题,优化代码,提高工作效率。
### Java毕业设计&课设——Spring源码深度解析+注解开发全套视频教程知识点概览 #### 一、Spring框架简介 Spring框架是由Rod Johnson创建的一款开源轻量级Java EE应用框架,旨在简化企业级应用的开发过程。Spring...
Spring框架是Java开发中最常用的...对于初学者,通过阅读和分析这些源码,可以加深对Spring的理解,提升开发技能。同时,文档则可能包含Spring的使用指南、最佳实践以及常见问题解答,是学习和解决问题的重要参考资料。
实践是检验真理的唯一标准,通过分析和运行源码,你将能更好地掌握Spring框架的精髓。 总之,这份资料对于Spring初学者或希望深入理解Spring的开发者来说是一份宝贵的资源。通过系统学习并结合实践,你将能够熟练地...
在本系列的第一篇中,我们将深入探讨Spring框架的核心特性——依赖注入...手写DI源码不仅有助于提高对Spring源码的理解,也有助于提升我们对软件设计原则和模式的认识,这对于成为一名优秀的Java开发者至关重要。
通过深入学习Spring源码,不仅能提升我们的技术水平,还能培养良好的编程习惯和架构思维。 总之,这份资料包为初学者提供了一个很好的平台,通过实例配置文件和源码,我们可以深入理解Spring的基础知识,包括依赖...
Spring框架的源代码分析主要集中在它的核心特性——控制反转(IOC)容器上。IOC容器是Spring的核心组件,它负责管理应用程序中的对象,也就是所谓的"bean"。BeanFactory接口是IOC容器的基础,它定义了最基础的bean...
在本篇中,我们将深入探讨Spring框架的核心组件——IoC(Inversion of Control)容器,这是Spring 2.5.6版本的一部分。IoC容器是Spring框架的心脏,它负责管理对象的生命周期和依赖关系,使得开发者能够实现松耦合和...
### Spring源码分析 #### 1. **Spring核心容器** - **BeanFactory**:这是Spring中最基本的接口,定义了如何管理和控制Bean的生命周期。 - **ApplicationContext**:它是BeanFactory的子接口,提供了更多的功能,...
### Spring源码解析:IOC容器 #### 一、Spring框架简介 Spring框架是一个开源的轻量级企业级应用开发框架,旨在简化企业级应用的开发过程。它通过一系列的组件和设计模式支持如依赖注入(Dependency Injection, DI...