spring 加載bean過程源碼簡易解剖

這一篇主要是講用載入bean的過程。其實就是IOC.低調 低調。。 

我把重要的都挑出來了。一步步往下看就明白spring載入bean.xml裏面bean的原理 。 
感受像候傑的 MFC深刻淺出,哈哈。 

觀看規則 

接下 表示下一層代碼。 
接上 表示最近上面要調用的代碼的詳細部分。 

node

Java代碼  收藏代碼spring

  1. public class XmlBeanFactory extends DefaultListableBeanFactory {  dom

  2.   

  3.         //新建一個bean分析器,把this註冊到裏面是由於,在分析器解析好一個bean時,能夠當即用這個this裏的註冊方法去保存bean,往下看就明白。任何bean到最後都是保存在XmlBeanFactory裏的(實際上是DefaultListableBeanFactory)。   ui

  4.   

  5.     private final XmlBeanDefinitionReader reader = new XmlBeanDefinitionReader(this);  this

  6.   

  7.     public XmlBeanFactory(Resource resource, BeanFactory parentBeanFactory) throws BeansException {  spa

  8.   

  9.         super(parentBeanFactory);  xml

  10.                 //載入xml文件  索引

  11.         this.reader.loadBeanDefinitions(resource); //往下->  get

  12.     }  it

  13. }  

  14.   

  15. public class XmlBeanDefinitionReader extends AbstractBeanDefinitionReader {  

  16.      //接上  

  17.      public int loadBeanDefinitions(Resource resource) throws BeansException {  

  18.   

  19.         InputStream is = null;  

  20.   

  21.         DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();  

  22.   

  23.         factory.setValidating(this.validating);  

  24.   

  25.         DocumentBuilder docBuilder = factory.newDocumentBuilder();  

  26.   

  27.         docBuilder.setErrorHandler(this.errorHandler);  

  28.   

  29.         if (this.entityResolver != null) {  

  30.             docBuilder.setEntityResolver(this.entityResolver);  

  31.         }  

  32.         is = resource.getInputStream();  

  33. //用Xerces解析xml,生成dom  

  34.         Document doc = docBuilder.parse(is);  

  35. //registerBeanDefinitions分析dom  

  36.         return registerBeanDefinitions(doc, resource); //往下  

  37.   

  38.     }  

  39.   

  40. //接上  

  41.         public int registerBeanDefinitions(Document doc, Resource resource) throws BeansException {  

  42.   

  43.         XmlBeanDefinitionParser parser = (XmlBeanDefinitionParser) BeanUtils.instantiateClass(this.parserClass);  

  44. //這個parserClass 是  DefaultXmlBeanDefinitionParser.class  

  45.         return parser.registerBeanDefinitions(this, doc, resource); //往下->  

  46.   

  47.     }  

  48.   

  49. }  

  50.   

  51. public class DefaultXmlBeanDefinitionParser implements XmlBeanDefinitionParser {  

  52. //明顯就是bean.xml裏面出現的很熟悉的標籤,說明已經快到底層類了  

  53.          

  54.   

  55.     public static final String AUTOWIRE_BY_NAME_VALUE = "byName";  

  56.     public static final String AUTOWIRE_BY_TYPE_VALUE = "byType";  

  57.     public static final String DEFAULT_LAZY_INIT_ATTRIBUTE = "default-lazy-init";  

  58.     public static final String DEFAULT_DEPENDENCY_CHECK_ATTRIBUTE = "default-dependency-check";  

  59.     public static final String DEFAULT_AUTOWIRE_ATTRIBUTE = "default-autowire";  

  60.     public static final String NAME_ATTRIBUTE = "name";  

  61.     public static final String ALIAS_ATTRIBUTE = "alias";  

  62.     public static final String BEAN_ELEMENT = "bean";  

  63.     public static final String ID_ATTRIBUTE = "id";  

  64.     public static final String PARENT_ATTRIBUTE = "parent";  

  65.     public static final String CLASS_ATTRIBUTE = "class";  

  66.     public static final String SINGLETON_ATTRIBUTE = "singleton";  

  67.     public static final String LAZY_INIT_ATTRIBUTE = "lazy-init";  

  68.     public static final String AUTOWIRE_ATTRIBUTE = "autowire";  

  69.     //...  

  70.    //接上  

  71.    public int registerBeanDefinitions(BeanDefinitionReader reader, Document doc, Resource resource) throws BeanDefinitionStoreException {  

  72.   

  73.         this.beanDefinitionReader = reader;  

  74.         this.resource = resource;;  

  75.         Element root = doc.getDocumentElement();  

  76.         //...  

  77.   

  78. //這裏準備開始正式解析bean  

  79.         int beanDefinitionCount = parseBeanDefinitions(root);//往下->  

  80. //這個beanDefinitionCount 就是解析出了多少個<bean></bean>  

  81.           

  82. //...  

  83.         return beanDefinitionCount;  

  84.     }  

  85.   

  86. protected int parseBeanDefinitions(Element root) throws BeanDefinitionStoreException {  

  87. //Xerces開始循環找<bean>標籤  

  88.         NodeList nl = root.getChildNodes();  

  89.         int beanDefinitionCounter = 0;  

  90.         for (int i = 0; i < nl.getLength(); i++) {  

  91.             Node node = nl.item(i);  

  92.             if (node instanceof Element) {  

  93.                 Element ele = (Element) node;  

  94.                 if // ...  

  95. //..  

  96.                 else if (BEAN_ELEMENT.equals(node.getNodeName())) {//這裏是重點,開始解析bean  

  97.                     beanDefinitionCounter++;  

  98. //分兩步走,看下面詳解。1.先把bean放到BeanDefinitionHolder  

  99.                     BeanDefinitionHolder bdHolder = parseBeanDefinitionElement(ele);//往下 1.->  

  100. //2.而後XmlBeanFactory去註冊  

  101.                     BeanDefinitionReaderUtils.registerBeanDefinition(  

  102.                             bdHolder, this.beanDefinitionReader.getBeanFactory()); //往下 2. ->  

  103.                 }  

  104.             }  

  105.         }  

  106.         return beanDefinitionCounter;  

  107.     }  

  108.   

  109. //接上1. 哈哈,下面是第一步,是正常解析bean,在同一個類中  

  110. protected BeanDefinitionHolder parseBeanDefinitionElement(Element ele) throws BeanDefinitionStoreException {  

  111.         //...  

  112.                 //下面能夠看到其實最底層的解析bean在同一個類的parseBeanDefinitionElement方法裏。由於spring把bean封裝成BeanDefinition 再把BeanDefinition 封裝成BeanDefinitionHolder   

  113.   

  114.         BeanDefinition beanDefinition = parseBeanDefinitionElement(ele, beanName);//往下  

  115.   

  116. //...  

  117.         return new BeanDefinitionHolder(beanDefinition, beanName, aliasesArray);  

  118.     }  

  119.   

  120. //接上 , 這個方法很長,畢竟<bean>裏attribute不少。  

  121.   protected BeanDefinition parseBeanDefinitionElement(Element ele, String beanName) throws BeanDefinitionStoreException {  

  122.         try {  

  123.                         //下面解析<bean>裏的<property>,這個我不分析了。  

  124.             MutablePropertyValues pvs = parsePropertyElements(ele, beanName);  

  125.                         //將BeanDefinition封裝成AbstractBeanDefinition   

  126.             AbstractBeanDefinition bd = BeanDefinitionReaderUtils.createBeanDefinition(  

  127.                     className, parent, cargs, pvs, this.beanDefinitionReader.getBeanClassLoader());  

  128.             //...         

  129.             return bd;  

  130.         }  

  131.         catch (/*...*/)  

  132.                    //...  

  133.   

  134.         }  

  135.     }  

  136.   

  137.   

  138. }  

  139. //bean解析部分到此結束。。。。  

  140.   

  141.   

  142. //接上2. 這裏是第二部,註冊部分,回到上面註釋裏的分兩部走這裏。  

  143. public class BeanDefinitionReaderUtils {  

  144.   public static void registerBeanDefinition(  

  145.             BeanDefinitionHolder bdHolder, BeanDefinitionRegistry beanFactory) throws BeansException {  

  146.   

  147.           

  148. //beanFactory就是XmlBeanFactory,實際上是它的父類 DefaultListableBeanFactory在執行registerBeanDefinition  

  149.   

  150.   

  151.         beanFactory.registerBeanDefinition(bdHolder.getBeanName(), bdHolder.getBeanDefinition()); //往下  

  152. //...  

  153.     }  

  154.   

  155. }  

  156.   

  157. public class DefaultListableBeanFactory extends AbstractAutowireCapableBeanFactory  

  158.     implements ConfigurableListableBeanFactory, BeanDefinitionRegistry {  

  159.     

  160.     /** Whether to allow re-registration of a different definition with the same name */  

  161.     private boolean allowBeanDefinitionOverriding = true;  

  162.   

  163.     /** Map of bean definition objects, keyed by bean name */  

  164. //下面是真正藏bean的地方,實際上是個Map,跟我預想的同樣。  

  165.     private final Map beanDefinitionMap = new HashMap();  

  166. //下面List多是給bean的名字作個索引,這是個人初步猜測。  

  167.     /** List of bean definition names, in registration order */  

  168.     private final List beanDefinitionNames = new ArrayList();  

  169.   //接上  

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

  171. //...  

  172.   Object oldBeanDefinition = this.beanDefinitionMap.get(beanName);  

  173.         if (oldBeanDefinition != null) {  

  174. //根據allowBeanDefinitionOverriding這個變量來決定在bean.xml裏的bean萬一有同名的狀況下否覆蓋,由於allowBeanDefinitionOverriding默認是true,因此覆蓋。  

  175.             if (!this.allowBeanDefinitionOverriding) {  

  176.                 throw new BeanDefinitionStoreException(...);  

  177.             }  

  178.             else {  

  179.                 //...只用註釋提醒相同bean將要被覆蓋了  

  180.             }  

  181.         }  

  182.         else {  

  183.                         //索引List里加上這個bean名字  

  184.             this.beanDefinitionNames.add(beanName);  

  185.         }  

  186.                 //將bean藏在map裏。用名字來索引。  

  187.         this.beanDefinitionMap.put(beanName, beanDefinition);  

  188.           

  189.     }  

  190. //...  

  191.   

  192. }  

  193. //結束  



能夠看到其實spring就是把bean.xml解析到一個map裏。 
至於獲取bean的方法 ,我不用說你們都知道了,到了底層就是 map.get("bean name"); 
萬流歸終啊。。。。 
over

相關文章
相關標籤/搜索