`
hotforcc
  • 浏览: 60648 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

Spring DispatcherServlet MVC 源码分析(BeanFactory的初始化)

阅读更多

BeanFactory的初始化的触发——><o:p></o:p>

FrameworkServlet中的wac.refresh()<o:p></o:p>

<o:p> </o:p>
如下为初始化
MVC部分的操作:<o:p></o:p>

java 代码
  1. initmultipartresolver();  
  2. initlocaleresolver();  
  3. initthemeresolver();  
  4. inithandlermappings();  
  5. inithandleradapters();  
  6. inithandlerexceptionresolvers();  
  7. initviewresolver();  

<o:p></o:p>

<o:p> </o:p>

现在有一个疑问 initHandlerMappings(ApplicationContext context)中的ApplicationContext的实现在web中是哪一个,可以从下面得到结论<o:p></o:p>

<o:p> </o:p>

java 代码
 
  1. public static final Class DEFAULT_CONTEXT_CLASS = XmlWebApplicationContext.class;  
  2. private Class contextClass = DEFAULT_CONTEXT_CLASS;  

<o:p></o:p>

<o:p> </o:p>

可以看出这个地方是实例化的是XmlWebApplicationContext的一个实例,那么这个实例是在哪个地方被实例化的呢?在DispatchServelet的父类FrameworkServlet中可以找到如下的代码,<o:p></o:p>

<o:p> </o:p>

java 代码
  1. WebApplicationContext parent = WebApplicationContextUtils.getWebApplicationContext(getServletContext());  
  2. WebApplicationContext wac = createWebApplicationContext(parent);  

<o:p></o:p>

<o:p> </o:p>

ok,我们现在就可以用这个wac了,顺便看看这个实例里边都有什么东西:<o:p></o:p>

<o:p> </o:p>

java 代码
 
  1. wac.setParent(parent);//web.xml中加载的context  
  2. wac.setServletContext(getServletContext());  
  3. wac.setServletConfig(getServletConfig());  
  4. wac.setNamespace(getNamespace());  
  5. if (getContextConfigLocation() != null) {  
  6.     wac.setConfigLocations(  
  7.     StringUtils.tokenizeToStringArray(  
  8.     getContextConfigLocation(), ConfigurableWebApplicationContext.CONFIG_LOCATION_DELIMITERS));  
  9. }//比较有用  
  10. wac.addApplicationListener(this);  

<o:p></o:p>

<o:p> </o:p>

继续我们的MVC探索--><o:p></o:p>

下面是MVC的部分,提前分析一下<o:p></o:p>

java 代码
 
  1. protected List getDefaultStrategies(ApplicationContext context, Class strategyInterface) throws BeansException {  
  2.     String key = strategyInterface.getName();//equals org.springframework.web.servlet.HandlerMapping  
  3.     List strategies = null;  
  4.     String value = defaultStrategies.getProperty(key);  
  5.     //org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping.class  
  6.     if (value != null) {  
  7.         String[] classNames = StringUtils.commaDelimitedListToStringArray(value);//等效于split  
  8.         strategies = new ArrayList(classNames.length);  
  9.         for (int i = 0; i < classNames.length; i++) {  
  10.             String className = classNames[i];  
  11.             try {  
  12.                 Class clazz = ClassUtils.forName(className, getClass().getClassLoader());  
  13.                 Object strategy = createDefaultStrategy(context, clazz);  
  14.                 strategies.add(strategy);  
  15.             }  
  16.             catch (ClassNotFoundException ex) {  
  17.                 throw new BeanInitializationException(  
  18.                     "Could not find DispatcherServlet's default strategy class [" + className +  
  19.                     "] for interface [" + key + "]", ex);  
  20.             }  
  21.             catch (LinkageError err) {  
  22.                 throw new BeanInitializationException(  
  23.                     "Error loading DispatcherServlet's default strategy class [" + className +  
  24.                     "] for interface [" + key + "]: problem with class file or dependent class", err);  
  25.             }  
  26.         }  
  27.     }else {  
  28.         strategies = Collections.EMPTY_LIST;  
  29.     }  
  30.     return strategies;  
  31. }  
  32.   
  33. protected Object createDefaultStrategy(ApplicationContext context, Class clazz) throws BeansException {  
  34.     return context.getAutowireCapableBeanFactory().createBean(  
  35.         clazz, AutowireCapableBeanFactory.AUTOWIRE_NO, false);  
  36.         //org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping.class  
  37. }  

<o:p></o:p>

context XmlWebApplicationContext的一个实例,而<o:p></o:p>

<o:p> </o:p>

java 代码
 
  1. public class XmlWebApplicationContext extends AbstractRefreshableWebApplicationContext  
  2.   
  3. public abstract class AbstractRefreshableWebApplicationContext extends AbstractRefreshableApplicationContext  
  4. implements ConfigurableWebApplicationContext, ThemeSource  
  5.   
  6. public abstract class AbstractRefreshableApplicationContext extends AbstractApplicationContext   
  7.   
  8. public abstract class AbstractApplicationContext extends DefaultResourceLoader  
  9. implements ConfigurableApplicationContext, DisposableBean {  
  10.     public AutowireCapableBeanFactory getAutowireCapableBeanFactory() throws IllegalStateException {  
  11.         return getBeanFactory();  
  12.     }  
  13. }  
  14.   
  15. 在AbstractRefreshableApplicationContext中  
  16. public final ConfigurableListableBeanFactory getBeanFactory() {  
  17.     synchronized (this.beanFactoryMonitor) {  
  18.         if (this.beanFactory == null) {  
  19.             throw new IllegalStateException("BeanFactory not initialized or already closed - " +  
  20.             "call 'refresh' before accessing beans via the ApplicationContext");  
  21.         }  
  22.         return this.beanFactory;  
  23.     }  
  24. }  

<o:p>

说明这个时候BeanFactory已经初始化好了,那么在哪个地方初始化好的呢?可以想象到的地方就是XmlWebApplicationContextnew的时候会初始化这个BeanFactory?在WebApplicationContext createWebApplicationContext(WebApplicationContext parent)时候,也就是在初始化ApplicationContext的时候有如下操作<o:p></o:p>

<o:p> </o:p>

wac.refresh();<o:p></o:p>

<o:p> </o:p>

也就是初始化ApplicationContext之后会马上初始化BeanFacory<o:p></o:p>

从类结构里我们能找到这个方法来自它的父类: AbstractApplicationContext 在它的 refresh() 方法内我们可以看到 spring 的复杂逻辑。首先执行了refreshBeanFactory(); (来自 AbstractRefreshableApplicationContext )见 (a),<o:p></o:p>

<o:p> </o:p>

(a)refreshBeanFactory(); 这个方法由负责维护变量 beanFactory 的子类AbstractRefreshableApplicationContext 实现,默认情况下<o:p></o:p>

这个方法直接实例化一个新的 DefaultListableBeanFactory 类型的 BeanFacorty, <o:p></o:p>

java 代码
 
  1. protected final void refreshBeanFactory() throws BeansException {  
  2.     DefaultListableBeanFactory beanFactory = createBeanFactory();  
  3.     customizeBeanFactory(beanFactory);  
  4.     loadBeanDefinitions(beanFactory);  
  5. }  
  6. protected DefaultListableBeanFactory createBeanFactory() {  
  7.     return new DefaultListableBeanFactory(getInternalParentBeanFactory());  
  8. }  
  9.   
  10. AbstractApplicationContext.java  
  11. protected BeanFactory getInternalParentBeanFactory() {//判断用的是ApplicationContext or BeanFactory   
  12.     return (getParent() instanceof ConfigurableApplicationContext) ?  
  13.         ((ConfigurableApplicationContext) getParent()).getBeanFactory() : (BeanFactory) getParent();  
  14. }  
  15. public class DefaultListableBeanFactory extends AbstractAutowireCapableBeanFactory  
  16. implements ConfigurableListableBeanFactory, BeanDefinitionRegistry {}  
  17.   
  18. XmlWebApplicationContext.java-->  
  19. /** 
  20. * Loads the bean definitions via an XmlBeanDefinitionReader. 
  21. * @see org.springframework.beans.factory.xml.XmlBeanDefinitionReader 
  22. * @see #initBeanDefinitionReader 
  23. * @see #loadBeanDefinitions 
  24. */  
  25. protected void loadBeanDefinitions(DefaultListableBeanFactory beanFactory) throws IOException {  
  26.     // Create a new XmlBeanDefinitionReader for the given BeanFactory.  
  27.     XmlBeanDefinitionReader beanDefinitionReader = new XmlBeanDefinitionReader(beanFactory);  
  28.   
  29.     // Configure the bean definition reader with this context's  
  30.     // resource loading environment.  
  31.     beanDefinitionReader.setResourceLoader(this);  
  32.     beanDefinitionReader.setEntityResolver(new ResourceEntityResolver(this));  
  33.   
  34.     // Allow a subclass to provide custom initialization of the reader,  
  35.     // then proceed with actually loading the bean definitions.  
  36.     initBeanDefinitionReader(beanDefinitionReader);  
  37.     loadBeanDefinitions(beanDefinitionReader);  
  38. }  
  39. public class XmlBeanDefinitionReader extends AbstractBeanDefinitionReader {  
  40. }  
  41. XmlBeanDefinitionReader 继承了 AbstractBeanDefinitionReader,AbstractBeanDefinitionReader的构造函数如下:  
  42. protected AbstractBeanDefinitionReader(BeanDefinitionRegistry beanFactory) {//DefaultListableBeanFactory实现了  
  43.                                         //BeanDefinitionRegistry   
  44.     Assert.notNull(beanFactory, "Bean factory must not be null");  
  45.     this.beanFactory = beanFactory;  
  46.     // Determine ResourceLoader to use.如果beanFactory不但实现了BeanDefinitionRegistry,而且实现了ResourceLoader   
  47.     //通常的这样的情况是实现了org.springframework.context.ApplicationContext的BeanFactory,这一点可以查看(c)部分  
  48.     if (this.beanFactory instanceof ResourceLoader) {  
  49.         this.resourceLoader = (ResourceLoader) this.beanFactory;  
  50.     }else {  
  51.         this.resourceLoader = new PathMatchingResourcePatternResolver();  
  52.     }  
  53. }  

<o:p></o:p>

初始化完了AbstractBeanDefinitionReader之后,继续XmlBeanDefinitionReaderd的初始化:<o:p></o:p>

java 代码
 
  1. public XmlBeanDefinitionReader(BeanDefinitionRegistry beanFactory) {  
  2.     if (getResourceLoader() != null) {//getResourceLoader已经得到  
  3.         this.entityResolver = new ResourceEntityResolver(getResourceLoader());  
  4.         //entityResolver  
  5.     }else {  
  6.         this.entityResolver = new DelegatingEntityResolver(ClassUtils.getDefaultClassLoader());  
  7.     }  
  8. }  

<o:p></o:p>

上面的初始化相当于初始化了 XmlBeanDefinitionReaderresourceLoaderentityResolver<o:p></o:p>

然后就是对各个配置路径的初始化工作:<o:p></o:p>

java 代码
 
  1. protected void loadBeanDefinitions(XmlBeanDefinitionReader reader) throws BeansException, IOException {  
  2.     String[] configLocations = getConfigLocations();  
  3.     if (configLocations != null) {  
  4.         for (int i = 0; i < configLocations.length; i++) {  
  5.             reader.loadBeanDefinitions(configLocations[i]);  
  6.         }  
  7.     }  
  8. }  

<o:p></o:p>

XmlBeanDefinitionReader中的初始化工作,因此所有的xml的解析实际上是在这个类文件中完成的 <o:p></o:p>

java 代码
 
  1. public int loadBeanDefinitions(EncodedResource encodedResource) throws BeanDefinitionStoreException {  
  2.     InputStream inputStream = encodedResource.getResource().getInputStream();  
  3.     try {  
  4.         InputSource inputSource = new InputSource(inputStream);  
  5.         if (encodedResource.getEncoding() != null) {  
  6.             inputSource.setEncoding(encodedResource.getEncoding());  
  7.         }  
  8.         return doLoadBeanDefinitions(inputSource, encodedResource.getResource());  
  9.         }  
  10.         finally{  
  11.             inputStream.close();  
  12.         }  
  13.     }catch (IOException ex) {  
  14.         throw new BeanDefinitionStoreException(  
  15.         "IOException parsing XML document from " + encodedResource.getResource(), ex);  
  16.     }  
  17. }  
  18. protected int doLoadBeanDefinitions(InputSource inputSource, Resource resource)  
  19.     throws BeanDefinitionStoreException {  
  20.     int validationMode = getValidationModeForResource(resource);  
  21.     Document doc = this.documentLoader.loadDocument(//获得解析的dom路径  
  22.         inputSource, this.entityResolver, this.errorHandler, validationMode, this.namespaceAware);  
  23.     return registerBeanDefinitions(doc, resource);  
  24. }  
  25.   
  26. (c)  
  27. public abstract class AbstractApplicationContext extends DefaultResourceLoader  
  28. implements ConfigurableApplicationContext, DisposableBean{};//AbstractApplicationContext继承了DefaultResourceLoader  
  29. public class DefaultResourceLoader implements ResourceLoader{}//DefaultResourceLoader实现了ResourceLoader  
  30. public DefaultResourceLoader() {//构造函数  
  31.         this.classLoader = ClassUtils.getDefaultClassLoader();  
  32. }  
  33. public static ClassLoader getDefaultClassLoader() {   
  34.     cl = Thread.currentThread().getContextClassLoader();//取得Application级的ClassLoader      
  35. }  

<o:p></o:p>

<o:p> </o:p>

<o:p> </o:p>

然后调用一个起缓冲作用的配置函数生成一个将 beanFacroty 包装起来的对象 beanDefinitionReader ,然后对这个对象进行属性配置,实际上该方法主要负责生成一个临时的操作对象,对应调用的函数为“loadBeanDefinitions(beanFactory);”该方法为初始化期间较为重要的一个。该方法来自其子类:AbstractRefreshableWebApplicationContext <o:p></o:p>

<o:p>

java 代码
 
  1. protected void loadBeanDefinitions(XmlBeanDefinitionReader reader) throws BeansException, IOException {  
  2.     String[] configLocations = getConfigLocations();// xml文件的配置的路径  
  3.     if (configLocations != null) {  
  4.         for (int i = 0; i < configLocations.length; i++) {  
  5.             reader.loadBeanDefinitions(configLocations[i]);  
  6.         }  
  7.     }  
  8. }  

<o:p>

对应的函数:<o:p></o:p>

protected void loadBeanDefinitions(DefaultListableBeanFactory) ,然后这里又调用了自己定义的 <o:p></o:p>

protected void loadBeanDefinitions(XmlBeanDefinitionReader) 方法。此时,它就使用到了在以前中设置了的( wac.setConfigLocations(……)) 我们开发中密切相关的配置文件。(同时也要记住此时这个函数的参数 beanDefinitionReader ,之前已经设置了”beanDefinitionReader.setResourceLoader(this) “这里的 this 是我们在前面见到的 XmlWebApplicationContext (一个定义好了的上下文))。接着往下:reader.loadBeanDefinitions(configLocations[i]); reader 开始加载我们配置文件内的东西了,不过真正复杂的实现此时才开始,我们继续往下走,在接下来的方法内默认情况下会执行:if (resourceLoader instanceof ResourcePatternResolver)(该判断条件为true ,由于从上面我们知道: beanDefinitionReader.setResourceLoader(this); this 的类型为: XmlWebApplicationContext所以 ((ResourcePatternResolver) resourceLoader).getResources(location); 得到一个 Resource[] 数组,接下来调用:int loadCount = loadBeanDefinitions(resources); 该函数继续调用自己子类定义的一系列临时接口最终执行到 return doLoadBeanDefinitions(inputSource, encodedResource.getResource()); 在这个函数内初始化了处理 xml 文件的一些对象并将用户的配置文件解析为一个 Document 对象。然后又执行了一系列函数直到return parser.registerBeanDefinitions(this, doc, resource); 这个函数来自我们新建的 DefaultXmlBeanDefinitionParser,在这个类里最终执行了对 xml 文件的解析工作和对 beanFacroty 变量执行了设置工作。<o:p></o:p>

<o:p> </o:p>

(b)终于我们从这些繁杂的逻辑中跳了出来,继续执行 AbstractApplicationContext.refresh() 下面的工作,后续的代码主要仍旧是往一些常量里面设值。<o:p></o:p>

<o:p> </o:p>

此时 Spring BeanFactory初始化过程就结束了。
分享到:
评论

相关推荐

    Spring源码学习九:DispatcherServlet初始化源码分析1

    Spring源码学习九:DispatcherServlet初始化源码分析1

    spring_MVC源码

    弃用了struts,用spring mvc框架做了几个项目,感觉都不错,而且使用了注解方式,可以省掉一大堆配置文件。本文主要介绍使用注解方式配置的spring mvc,之前写的spring3.0 mvc和rest小例子没有介绍到数据层的内容,...

    Spring MVC 教程 快速入门 深入分析

    Spring MVC 教程 快速入门 深入分析 目录 一、前言 二、spring mvc 核心类与接口 三、spring mvc 核心流程图 四、spring mvc DispatcherServlet说明 五、spring mvc 双亲上下文的说明 六、springMVC-mvc.xml 配置...

    Spring Web MVC入门教程

    包括:功能、配置、上下文关系、初始化顺序等 第三章:注解式控制器开发详解 包括:注解式开发HelloWorld、处理器定义、请求映射、REST风格支持、请求数据映射、数据绑定等诸多内容 第四章:数据类型转换 包括:...

    Spring MVC入门教程

    四、spring mvc DispatcherServlet说明 五、spring mvc 双亲上下文的说明 六、springMVC-mvc.xml 配置文件片段讲解 七、spring mvc 如何访问到静态的文件,如jpg,js,css? 八、spring mvc 请求如何映射到具体的...

    spring-webmvc5.3.6 jar包.rar

    这个jar文件包含Spring MVC框架相关的所有类。  包含国际化、标签、Theme、视图展现的FreeMarker、JasperReports、Tiles、Velocity、XSLT相关类。  当然,如果你的应用使用了独立的MVC框架,则无需这个JAR文件里的...

    spring MVC 配置文档

    DispatcherServlet 是Spring MVC 的入口 所有进入Spring Web 的 Request 都经过 DispatcherServlet 需要在 web.xml 中注册 DispatcherServlet &lt;servlet&gt; &lt;servlet-name&gt;dispatherContext&lt;/servlet-name&gt; ...

    SpringMVC框架架构介绍

    四、spring mvc DispatcherServlet说明 五、spring mvc 双亲上下文的说明 六、springMVC-mvc.xml 配置文件片段讲解 七、spring mvc 如何访问到静态的文件,如jpg,js,css? 八、spring mvc 请求如何映射到具体的...

    Spring MVC 入门实例

    首先, 我需要在你心里建立起 Spring MVC 的基本概念. 基于 Spring 的 Web 应用程序接收到 http://localhost:8080/hello.do(事实上请求路径是 /hello.do) 的请求后, Spring 将这个请求交给一个名为 helloController ...

    Spring_MVC_教程_快速入门_深入分析

    spring mvc DispatcherServlet说明 spring mvc 双亲上下文的说明 springMVC-mvc.xml 配置文件片段讲解 spring mvc 如何访问到静态的文件,如jpg,js,css? spring mvc 处理ajax请求 spring mvc 转发与重定向 如何给...

    spring-mvc与xfire的集成

    spring-mvc与xfire的集成 对于这样的集成,看看里面的配置文件就行了 一个简单的接口,对应一个简单的实现。 然后在配置文件里指明就可以了。 DispatcherServlet本身会管理xfire的请求 配置文件中配置的key可以理解...

    springmvc demo

    Spring Web MVC是一种基于Java的实现了Web MVC设计模式的请求驱动类型的轻量级Web框架,即使用了MVC架构模式的思想,将web层进行职责解耦,基于请求驱动指的就是使用请求-响应模型,框架的目的就是帮助我们简化开发...

    spring-webmvc-portlet-4.3.14.RELEASE.jar

    Spring Portlet MVC和其Web MVC可以说是如出一辙,只是在Web MVC中处于核心的DispatcherServlet在Portlet MVC中换成了DispatcherPortlet

    Spring源码学习十:DispatcherServlet请求分发源码分析1

    总结:首先,SpringMVC框架在启动的时候会遍历Spring容器中的所有bean,对标注了@Controller或并,使用@RequestMapping注解

    Pro Spring MVC With Web Flow

    What you'll learn Key Spring Framework fundamentals How to use the Spring MVC architecture How to develop with the DispatcherServlet How to write Controllers How to work with Resolving and ...

    Spring MVC之DispatcherServlet详解_动力节点Java学院整理

    DispatcherServlet是前端控制器设计模式的实现,提供Spring Web MVC的集中访问点,而且负责职责的分派,而且与Spring IoC容器无缝集成,从而可以获得Spring的所有好处。 具体请参考第二章的图2-1。  ...

    Spring3 MVC结构全解(实例+核心流程+全部jar包)

    Spring3 MVC结构核心流程全解 全部jar包都是自己一个个找出来的 使用Spring MVC,配置DispatcherServlet

    全面掌握Spring MVC:从基础到高级的实践指南

    本文通过分析Spring MVC的核心组件和执行流程,提供了一个全面的学习指南。 Spring MVC基于Model-View-Controller(MVC)架构模式,优化了Web应用程序的设计和开发。在Spring MVC中,DispatcherServlet作为前端控制...

    spring MVC HelloWorld

    我自己写的Spring MVC的一个HelloWold,从建HelloWorld工程,搭建环境Install spring,配置web.xml,配置SpringMVC的配置文件springDispatcherServlet-servlet.xml,添加控制器类,建jsp页面,部署 Tomcat,最后运行...

    精通Spring.MVC

    中文名: 精通Spring MVC 原名: Pro Spring MVC With Web Flow 作者: Deinum Serneels Yates Ladd Vanfleteren图书分类: 软件 资源格式: PDF 版本: 英文版 出版社: Apress书号: 978-1-4302-4155-3发行时间: 2012年 ...

Global site tag (gtag.js) - Google Analytics