这一篇主要是讲用载入bean的过程。其实就是IOC.低调 低调。。

我把重要的都挑出来了。一步步往下看就明白spring载入bean.xml里面bean的原理 。 
感觉像候杰的 MFC深入浅出,哈哈。

观看规则

接下 表示下一层代码。 
接上 表示最近上面要调用的代码的详细部分。

  1. public class XmlBeanFactory extends DefaultListableBeanFactory {
  2. //新建一个bean分析器,把this注册到里面是因为,在分析器解析好一个bean时,可以立即用这个this里的注册方法去保存bean,往下看就明白。任何bean到最后都是保存在XmlBeanFactory里的(其实是DefaultListableBeanFactory)。
  3. private final XmlBeanDefinitionReader reader = new XmlBeanDefinitionReader(this);
  4. public XmlBeanFactory(Resource resource, BeanFactory parentBeanFactory) throws BeansException {
  5. super(parentBeanFactory);
  6. //载入xml文件
  7. this.reader.loadBeanDefinitions(resource); //往下->
  8. }
  9. }
  10. public class XmlBeanDefinitionReader extends AbstractBeanDefinitionReader {
  11. //接上
  12. public int loadBeanDefinitions(Resource resource) throws BeansException {
  13. InputStream is = null;
  14. DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
  15. factory.setValidating(this.validating);
  16. DocumentBuilder docBuilder = factory.newDocumentBuilder();
  17. docBuilder.setErrorHandler(this.errorHandler);
  18. if (this.entityResolver != null) {
  19. docBuilder.setEntityResolver(this.entityResolver);
  20. }
  21. is = resource.getInputStream();
  22. //用Xerces解析xml,生成dom
  23. Document doc = docBuilder.parse(is);
  24. //registerBeanDefinitions分析dom
  25. return registerBeanDefinitions(doc, resource); //往下
  26. }
  27. //接上
  28. public int registerBeanDefinitions(Document doc, Resource resource) throws BeansException {
  29. XmlBeanDefinitionParser parser = (XmlBeanDefinitionParser) BeanUtils.instantiateClass(this.parserClass);
  30. //这个parserClass 是  DefaultXmlBeanDefinitionParser.class
  31. return parser.registerBeanDefinitions(this, doc, resource); //往下->
  32. }
  33. }
  34. public class DefaultXmlBeanDefinitionParser implements XmlBeanDefinitionParser {
  35. //明显就是bean.xml里面出现的很熟悉的标签,说明已经快到底层类了
  36. public static final String AUTOWIRE_BY_NAME_VALUE = "byName";
  37. public static final String AUTOWIRE_BY_TYPE_VALUE = "byType";
  38. public static final String DEFAULT_LAZY_INIT_ATTRIBUTE = "default-lazy-init";
  39. public static final String DEFAULT_DEPENDENCY_CHECK_ATTRIBUTE = "default-dependency-check";
  40. public static final String DEFAULT_AUTOWIRE_ATTRIBUTE = "default-autowire";
  41. public static final String NAME_ATTRIBUTE = "name";
  42. public static final String ALIAS_ATTRIBUTE = "alias";
  43. public static final String BEAN_ELEMENT = "bean";
  44. public static final String ID_ATTRIBUTE = "id";
  45. public static final String PARENT_ATTRIBUTE = "parent";
  46. public static final String CLASS_ATTRIBUTE = "class";
  47. public static final String SINGLETON_ATTRIBUTE = "singleton";
  48. public static final String LAZY_INIT_ATTRIBUTE = "lazy-init";
  49. public static final String AUTOWIRE_ATTRIBUTE = "autowire";
  50. //...
  51. //接上
  52. public int registerBeanDefinitions(BeanDefinitionReader reader, Document doc, Resource resource) throws BeanDefinitionStoreException {
  53. this.beanDefinitionReader = reader;
  54. this.resource = resource;;
  55. Element root = doc.getDocumentElement();
  56. //...
  57. //这里准备开始正式解析bean
  58. int beanDefinitionCount = parseBeanDefinitions(root);//往下->
  59. //这个beanDefinitionCount 就是解析出了多少个<bean></bean>
  60. //...
  61. return beanDefinitionCount;
  62. }
  63. protected int parseBeanDefinitions(Element root) throws BeanDefinitionStoreException {
  64. //Xerces开始循环找<bean>标签
  65. NodeList nl = root.getChildNodes();
  66. int beanDefinitionCounter = 0;
  67. for (int i = 0; i < nl.getLength(); i++) {
  68. Node node = nl.item(i);
  69. if (node instanceof Element) {
  70. Element ele = (Element) node;
  71. if // ...
  72. //..
  73. else if (BEAN_ELEMENT.equals(node.getNodeName())) {//这里是重点,开始解析bean
  74. beanDefinitionCounter++;
  75. //分两步走,看下面详解。1.先把bean放到BeanDefinitionHolder
  76. BeanDefinitionHolder bdHolder = parseBeanDefinitionElement(ele);//往下 1.->
  77. //2.然后XmlBeanFactory去注册
  78. BeanDefinitionReaderUtils.registerBeanDefinition(
  79. bdHolder, this.beanDefinitionReader.getBeanFactory()); //往下 2. ->
  80. }
  81. }
  82. }
  83. return beanDefinitionCounter;
  84. }
  85. //接上1. 哈哈,下面是第一步,是正常解析bean,在同一个类中
  86. protected BeanDefinitionHolder parseBeanDefinitionElement(Element ele) throws BeanDefinitionStoreException {
  87. //...
  88. //下面可以看到其实最底层的解析bean在同一个类的parseBeanDefinitionElement方法里。因为spring把bean封装成BeanDefinition 再把BeanDefinition 封装成BeanDefinitionHolder
  89. BeanDefinition beanDefinition = parseBeanDefinitionElement(ele, beanName);//往下
  90. //...
  91. return new BeanDefinitionHolder(beanDefinition, beanName, aliasesArray);
  92. }
  93. //接上 , 这个方法很长,毕竟<bean>里attribute很多。
  94. protected BeanDefinition parseBeanDefinitionElement(Element ele, String beanName) throws BeanDefinitionStoreException {
  95. try {
  96. //下面解析<bean>里的<property>,这个我不分析了。
  97. MutablePropertyValues pvs = parsePropertyElements(ele, beanName);
  98. //将BeanDefinition封装成AbstractBeanDefinition
  99. AbstractBeanDefinition bd = BeanDefinitionReaderUtils.createBeanDefinition(
  100. className, parent, cargs, pvs, this.beanDefinitionReader.getBeanClassLoader());
  101. //...
  102. return bd;
  103. }
  104. catch (/*...*/)
  105. //...
  106. }
  107. }
  108. }
  109. //bean解析部分到此结束。。。。
  110. //接上2. 这里是第二部,注册部分,回到上面注释里的分两部走这里。
  111. public class BeanDefinitionReaderUtils {
  112. public static void registerBeanDefinition(
  113. BeanDefinitionHolder bdHolder, BeanDefinitionRegistry beanFactory) throws BeansException {
  114. //beanFactory就是XmlBeanFactory,其实是它的父类 DefaultListableBeanFactory在执行registerBeanDefinition
  115. beanFactory.registerBeanDefinition(bdHolder.getBeanName(), bdHolder.getBeanDefinition()); //往下
  116. //...
  117. }
  118. }
  119. public class DefaultListableBeanFactory extends AbstractAutowireCapableBeanFactory
  120. implements ConfigurableListableBeanFactory, BeanDefinitionRegistry {
  121. /** Whether to allow re-registration of a different definition with the same name */
  122. private boolean allowBeanDefinitionOverriding = true;
  123. /** Map of bean definition objects, keyed by bean name */
  124. //下面是真正藏bean的地方,其实是个Map,跟我预想的一样。
  125. private final Map beanDefinitionMap = new HashMap();
  126. //下面List可能是给bean的名字做个索引,这是我的初步猜想。
  127. /** List of bean definition names, in registration order */
  128. private final List beanDefinitionNames = new ArrayList();
  129. //接上
  130. public void registerBeanDefinition(String beanName, BeanDefinition beanDefinition) throws BeanDefinitionStoreException {
  131. //...
  132. Object oldBeanDefinition = this.beanDefinitionMap.get(beanName);
  133. if (oldBeanDefinition != null) {
  134. //根据allowBeanDefinitionOverriding这个变量来决定在bean.xml里的bean万一有同名的情况下否覆盖,因为allowBeanDefinitionOverriding默认是true,所以覆盖。
  135. if (!this.allowBeanDefinitionOverriding) {
  136. throw new BeanDefinitionStoreException(...);
  137. }
  138. else {
  139. //...只用注释提醒相同bean将要被覆盖了
  140. }
  141. }
  142. else {
  143. //索引List里加上这个bean名字
  144. this.beanDefinitionNames.add(beanName);
  145. }
  146. //将bean藏在map里。用名字来索引。
  147. this.beanDefinitionMap.put(beanName, beanDefinition);
  148. }
  149. //...
  150. }
  151. //结束

可以看到其实spring就是把bean.xml解析到一个map里。

转载路径:http://hzieept.iteye.com/blog/748283

spring 加载bean过程源码简易解剖(转载)的更多相关文章

  1. Dubbo实践(六)Spring加载Bean流程

    根据上一小节对于spring扩展schema的介绍,大概可以猜到dubbo中相关的内容是如何实现的. 再来回顾Dubbo实践(一)中定义的dubbo-provider.xml: <?xml ve ...

  2. spring加载bean流程解析

    spring作为目前我们开发的基础框架,每天的开发工作基本和他形影不离,作为管理bean的最经典.优秀的框架,它的复杂程度往往令人望而却步.不过作为朝夕相处的框架,我们必须得明白一个问题就是sprin ...

  3. spring加载Bean的解析过程(二)

    1.例如: BeanFactory bf = new XmlBeanFactory(new ClassPathResource("spring.xml")); User user ...

  4. spring加载过程,源码带你理解从初始化到bean注入

    spring在容器启动时,容器正式初始化入口refresh()如下图 ①包括初始化FactoryBean.解析XML注册所有BeanDefinition信息  ②包括注册scope管理类  ③初始化单 ...

  5. spring加载bean实例化顺序

    问题来源: 有一个bean为 A,一个bean为B.想要A在容器实例化的时候的一个属性name赋值为B的一个方法funB的返回值. 如果只是在A里单纯的写着: private B b;private ...

  6. spring加载bean报错:expected at least 1 bean which qualifies as autowire candidate. Dependency annotations: {}

    看具体报错日志: 警告: Unable to proxy interface-implementing method [public final void cn.wlf.selection.proto ...

  7. CloudStack采用spring加载bean(cloud-framework-spring-module模块)

    CloudStackContextLoaderListener /* * Licensed to the Apache Software Foundation (ASF) under one * or ...

  8. Spring多种加载Bean方式简析

    1 定义bean的方式 常见的定义Bean的方式有: 通过xml的方式,例如: <bean id="dictionaryRelMap" class="java.ut ...

  9. sping加载bean都发生了些什么

    问题描述:使用@Autowired注入的类,没有实例化 //Controller @RequestMapping(value="/deepblue") @Controller pu ...

随机推荐

  1. js出现Syntax error on token "catch", Identifier expected

    本文转自:http://blog.csdn.net/u011159417/article/details/73916676 项目中需要使用jQuery,因此下载了jQuery的js包jquery-3. ...

  2. Python监控文件变化:watchdog

    Python监控文件变化有两种库:pyinotify和watchdog.pyinotify依赖于Linux平台的inotify,后者则对不同平台的的事件都进行了封装.也就是说,watchdog跨平台. ...

  3. SPFA 上手题 数 枚:

    1, HDU 1548 A strange lift :http://acm.hdu.edu.cn/showproblem.php?pid=1548 这道题可以灰常巧妙的转换为一道最短路题目,对第i层 ...

  4. Jsp应用EL和JSTL实例对比。

    普通方式: register.jsp <%@ page language="java" import="java.util.*" pageEncoding ...

  5. iPhone手机屏幕的尺寸180330更新

    以下是 iPhone的型号和对应的屏幕宽高 英寸  宽 高  厚度 3.5   320 480 4s      ipad   系列   4   320 568 5   5s   4.7  375 66 ...

  6. pythonl学习笔记——爬虫的基本常识

    1 robots协议 Robots协议(也称为爬虫协议.机器人协议等)的全称是“网络爬虫排除标准”(Robots Exclusion Protocol),网站通过Robots协议告诉搜索引擎哪些页面可 ...

  7. JMeter学习笔记--JMeter执行顺序规则

    JMeter执行顺序规则: 配置元件 前置处理器 定时器 采样器 后置处理器(除非服务器响应为空) 断言 监听器 只有当作用域内存在采样器时,定时器.断言.前置/后置处理器才会被执行,逻辑控制器和采样 ...

  8. Linux内核中锁机制之原子操作、自旋锁

    很多人会问这样的问题,Linux内核中提供了各式各样的同步锁机制到底有何作用?追根到底其实是由于操作系统中存在多进程对共享资源的并发访问,从而引起了进程间的竞态.这其中包括了我们所熟知的SMP系统,多 ...

  9. Bitmap具体解释与Bitmap的内存优化

    感觉这里的排版看着更舒服些 Bitmap具体解释与Bitmap的内存优化 一.Bitmap: Bitmap是Android系统中的图像处理的最重要类之中的一个.用它能够获取图像文件信息,进行图像剪切. ...

  10. JWPlayer高速入门指南(中文)

    将JW Player嵌入到网页中很的简单,仅仅须要进行例如以下3个步骤: 1.解压mediaplayer-viral.zip文件.将jwplayer.js和player.swf文件复制到project ...