spring 加载bean过程源码简易解剖(转载)
这一篇主要是讲用载入bean的过程。其实就是IOC.低调 低调。。
我把重要的都挑出来了。一步步往下看就明白spring载入bean.xml里面bean的原理 。
感觉像候杰的 MFC深入浅出,哈哈。
观看规则
接下 表示下一层代码。
接上 表示最近上面要调用的代码的详细部分。
- public class XmlBeanFactory extends DefaultListableBeanFactory {
- //新建一个bean分析器,把this注册到里面是因为,在分析器解析好一个bean时,可以立即用这个this里的注册方法去保存bean,往下看就明白。任何bean到最后都是保存在XmlBeanFactory里的(其实是DefaultListableBeanFactory)。
- private final XmlBeanDefinitionReader reader = new XmlBeanDefinitionReader(this);
- public XmlBeanFactory(Resource resource, BeanFactory parentBeanFactory) throws BeansException {
- super(parentBeanFactory);
- //载入xml文件
- this.reader.loadBeanDefinitions(resource); //往下->
- }
- }
- public class XmlBeanDefinitionReader extends AbstractBeanDefinitionReader {
- //接上
- public int loadBeanDefinitions(Resource resource) throws BeansException {
- InputStream is = null;
- DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
- factory.setValidating(this.validating);
- DocumentBuilder docBuilder = factory.newDocumentBuilder();
- docBuilder.setErrorHandler(this.errorHandler);
- if (this.entityResolver != null) {
- docBuilder.setEntityResolver(this.entityResolver);
- }
- is = resource.getInputStream();
- //用Xerces解析xml,生成dom
- Document doc = docBuilder.parse(is);
- //registerBeanDefinitions分析dom
- return registerBeanDefinitions(doc, resource); //往下
- }
- //接上
- public int registerBeanDefinitions(Document doc, Resource resource) throws BeansException {
- XmlBeanDefinitionParser parser = (XmlBeanDefinitionParser) BeanUtils.instantiateClass(this.parserClass);
- //这个parserClass 是 DefaultXmlBeanDefinitionParser.class
- return parser.registerBeanDefinitions(this, doc, resource); //往下->
- }
- }
- public class DefaultXmlBeanDefinitionParser implements XmlBeanDefinitionParser {
- //明显就是bean.xml里面出现的很熟悉的标签,说明已经快到底层类了
- public static final String AUTOWIRE_BY_NAME_VALUE = "byName";
- public static final String AUTOWIRE_BY_TYPE_VALUE = "byType";
- public static final String DEFAULT_LAZY_INIT_ATTRIBUTE = "default-lazy-init";
- public static final String DEFAULT_DEPENDENCY_CHECK_ATTRIBUTE = "default-dependency-check";
- public static final String DEFAULT_AUTOWIRE_ATTRIBUTE = "default-autowire";
- public static final String NAME_ATTRIBUTE = "name";
- public static final String ALIAS_ATTRIBUTE = "alias";
- public static final String BEAN_ELEMENT = "bean";
- public static final String ID_ATTRIBUTE = "id";
- public static final String PARENT_ATTRIBUTE = "parent";
- public static final String CLASS_ATTRIBUTE = "class";
- public static final String SINGLETON_ATTRIBUTE = "singleton";
- public static final String LAZY_INIT_ATTRIBUTE = "lazy-init";
- public static final String AUTOWIRE_ATTRIBUTE = "autowire";
- //...
- //接上
- public int registerBeanDefinitions(BeanDefinitionReader reader, Document doc, Resource resource) throws BeanDefinitionStoreException {
- this.beanDefinitionReader = reader;
- this.resource = resource;;
- Element root = doc.getDocumentElement();
- //...
- //这里准备开始正式解析bean
- int beanDefinitionCount = parseBeanDefinitions(root);//往下->
- //这个beanDefinitionCount 就是解析出了多少个<bean></bean>
- //...
- return beanDefinitionCount;
- }
- protected int parseBeanDefinitions(Element root) throws BeanDefinitionStoreException {
- //Xerces开始循环找<bean>标签
- NodeList nl = root.getChildNodes();
- int beanDefinitionCounter = 0;
- for (int i = 0; i < nl.getLength(); i++) {
- Node node = nl.item(i);
- if (node instanceof Element) {
- Element ele = (Element) node;
- if // ...
- //..
- else if (BEAN_ELEMENT.equals(node.getNodeName())) {//这里是重点,开始解析bean
- beanDefinitionCounter++;
- //分两步走,看下面详解。1.先把bean放到BeanDefinitionHolder
- BeanDefinitionHolder bdHolder = parseBeanDefinitionElement(ele);//往下 1.->
- //2.然后XmlBeanFactory去注册
- BeanDefinitionReaderUtils.registerBeanDefinition(
- bdHolder, this.beanDefinitionReader.getBeanFactory()); //往下 2. ->
- }
- }
- }
- return beanDefinitionCounter;
- }
- //接上1. 哈哈,下面是第一步,是正常解析bean,在同一个类中
- protected BeanDefinitionHolder parseBeanDefinitionElement(Element ele) throws BeanDefinitionStoreException {
- //...
- //下面可以看到其实最底层的解析bean在同一个类的parseBeanDefinitionElement方法里。因为spring把bean封装成BeanDefinition 再把BeanDefinition 封装成BeanDefinitionHolder
- BeanDefinition beanDefinition = parseBeanDefinitionElement(ele, beanName);//往下
- //...
- return new BeanDefinitionHolder(beanDefinition, beanName, aliasesArray);
- }
- //接上 , 这个方法很长,毕竟<bean>里attribute很多。
- protected BeanDefinition parseBeanDefinitionElement(Element ele, String beanName) throws BeanDefinitionStoreException {
- try {
- //下面解析<bean>里的<property>,这个我不分析了。
- MutablePropertyValues pvs = parsePropertyElements(ele, beanName);
- //将BeanDefinition封装成AbstractBeanDefinition
- AbstractBeanDefinition bd = BeanDefinitionReaderUtils.createBeanDefinition(
- className, parent, cargs, pvs, this.beanDefinitionReader.getBeanClassLoader());
- //...
- return bd;
- }
- catch (/*...*/)
- //...
- }
- }
- }
- //bean解析部分到此结束。。。。
- //接上2. 这里是第二部,注册部分,回到上面注释里的分两部走这里。
- public class BeanDefinitionReaderUtils {
- public static void registerBeanDefinition(
- BeanDefinitionHolder bdHolder, BeanDefinitionRegistry beanFactory) throws BeansException {
- //beanFactory就是XmlBeanFactory,其实是它的父类 DefaultListableBeanFactory在执行registerBeanDefinition
- beanFactory.registerBeanDefinition(bdHolder.getBeanName(), bdHolder.getBeanDefinition()); //往下
- //...
- }
- }
- public class DefaultListableBeanFactory extends AbstractAutowireCapableBeanFactory
- implements ConfigurableListableBeanFactory, BeanDefinitionRegistry {
- /** Whether to allow re-registration of a different definition with the same name */
- private boolean allowBeanDefinitionOverriding = true;
- /** Map of bean definition objects, keyed by bean name */
- //下面是真正藏bean的地方,其实是个Map,跟我预想的一样。
- private final Map beanDefinitionMap = new HashMap();
- //下面List可能是给bean的名字做个索引,这是我的初步猜想。
- /** List of bean definition names, in registration order */
- private final List beanDefinitionNames = new ArrayList();
- //接上
- public void registerBeanDefinition(String beanName, BeanDefinition beanDefinition) throws BeanDefinitionStoreException {
- //...
- Object oldBeanDefinition = this.beanDefinitionMap.get(beanName);
- if (oldBeanDefinition != null) {
- //根据allowBeanDefinitionOverriding这个变量来决定在bean.xml里的bean万一有同名的情况下否覆盖,因为allowBeanDefinitionOverriding默认是true,所以覆盖。
- if (!this.allowBeanDefinitionOverriding) {
- throw new BeanDefinitionStoreException(...);
- }
- else {
- //...只用注释提醒相同bean将要被覆盖了
- }
- }
- else {
- //索引List里加上这个bean名字
- this.beanDefinitionNames.add(beanName);
- }
- //将bean藏在map里。用名字来索引。
- this.beanDefinitionMap.put(beanName, beanDefinition);
- }
- //...
- }
- //结束
可以看到其实spring就是把bean.xml解析到一个map里。
转载路径:http://hzieept.iteye.com/blog/748283
spring 加载bean过程源码简易解剖(转载)的更多相关文章
- Dubbo实践(六)Spring加载Bean流程
根据上一小节对于spring扩展schema的介绍,大概可以猜到dubbo中相关的内容是如何实现的. 再来回顾Dubbo实践(一)中定义的dubbo-provider.xml: <?xml ve ...
- spring加载bean流程解析
spring作为目前我们开发的基础框架,每天的开发工作基本和他形影不离,作为管理bean的最经典.优秀的框架,它的复杂程度往往令人望而却步.不过作为朝夕相处的框架,我们必须得明白一个问题就是sprin ...
- spring加载Bean的解析过程(二)
1.例如: BeanFactory bf = new XmlBeanFactory(new ClassPathResource("spring.xml")); User user ...
- spring加载过程,源码带你理解从初始化到bean注入
spring在容器启动时,容器正式初始化入口refresh()如下图 ①包括初始化FactoryBean.解析XML注册所有BeanDefinition信息 ②包括注册scope管理类 ③初始化单 ...
- spring加载bean实例化顺序
问题来源: 有一个bean为 A,一个bean为B.想要A在容器实例化的时候的一个属性name赋值为B的一个方法funB的返回值. 如果只是在A里单纯的写着: private B b;private ...
- 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 ...
- CloudStack采用spring加载bean(cloud-framework-spring-module模块)
CloudStackContextLoaderListener /* * Licensed to the Apache Software Foundation (ASF) under one * or ...
- Spring多种加载Bean方式简析
1 定义bean的方式 常见的定义Bean的方式有: 通过xml的方式,例如: <bean id="dictionaryRelMap" class="java.ut ...
- sping加载bean都发生了些什么
问题描述:使用@Autowired注入的类,没有实例化 //Controller @RequestMapping(value="/deepblue") @Controller pu ...
随机推荐
- iOS开发之多媒体播放
iOS开发之多媒体播放 iOS sdk中提供了很多方便的方法来播放多媒体.本文将利用这些SDK做一个demo,来讲述一下如何使用它们来播放音频文件. AudioToolbox framework 使用 ...
- 设置当前Activity的屏幕亮度
设置当前的Activity的屏幕亮度,而不是设置系统的屏幕亮度,退出当前的Activity后恢复系统的亮度. 直接看代码好了 Java代码 WindowManager.LayoutParams lp ...
- sqlserver中将varchar类型转换为int型再进行排序的方法
sql中把varchar类型转换为int型然后进行排序,如果我们数据库的ID设置为varchar型的 在查询的时候order by id的话 如果我们数据库的ID设置为varchar型的 在查询的时候 ...
- Bridge - 桥接模式
1. 概述 在软件系统中,某些类型由于自身的逻辑,它具有两个或多个维度的变化,那么如何应对这种“多维度的变化”?如何利用面向对象的技术来使得该类型能够轻松的沿着多个方向进行变化,而又不引入额外的复杂度 ...
- [MFC]图形附加alpha透明通道
改动图形而且附加透明通道: 要附加透明度,能够要把图片转化为32位png图片,然后设置对应的alpha值: 1. 怎样把一张图片改动为32位的Png: a) 读取原图片颜色信息 ...
- Spring注解 强大了个强大--详解注解方式
好文章:http://tonyaction.blog.51cto.com/227462/83874/ http://www.ibm.com/developerworks/cn/java/j-lo-sp ...
- 使用matlab处理图像的基础知识
MATLAB基本函数一 矩阵运算 1.基本算数运算(加减乘除) + -运算要求矩阵维数相同,例m*n * /运算,例A=B*C,B矩阵是m*n矩阵,B是n*p矩阵,则A是m*p矩阵 A/B相当于A*i ...
- mysql-5.7 innodb_file_per_table 详解
一.innodb_file_per_table 的简要说明: 在很久很久以前也就是说还没有innodb_file_per_table 的那个年代,所有的innodb表的数据都是保存在innodb系统表 ...
- kafka工作原理简介
消息队列 消息队列技术是分布式应用间交换信息的一种技术.消息队列可驻留在内存或磁盘上, 队列存储消息直到它们被应用程序读走.通过消息队列,应用程序可独立地执行--它们不需要知道彼此的位置.或在继续执行 ...
- linux下man手册简介
Linux提供了丰富的帮助手册,当你需要查看某个命令的参数时不必到处上网查找,只要man一下即可.Linux 的man手册共有以下几个章节: 1.Standard commands (标准命令)2.S ...