记录并分享一下本人学习spring源码的过程,有什么问题或者补充会持续更新。欢迎大家指正!

环境: spring5.X + idea

Spring 是一个工厂,是一个负责对象的创建和维护的工厂。它给我提供了一个功能齐全而且方便我们使用的ApplicationContext子接口,它最底层的接口是BeanFactory。在这个BeanFactory下面衍生了各种功能的子接口。

  1. 容器管理HierarchicalBeanFatory
  2. 自动注入AutowireCapableBeanFactory
  3. 读取配置信息ListableBeanFactory

可以自行找一下BeanFactory类关系图,它有一个子实现类XmlBeanFactory,先说一下XML配置文件的读取

ListableBeanFactory 是读取配置信息的,它的子实现类XmlBeanFactory就是读取xml文件的具体实现。
而ApplicationContext继承了ListableBeanFactory并对xml解析做了进一步的封装所以再我们使用ApplicationContext
时直接给他一个对应位置的资源文件名它就会帮我们读取到配置信息。 ApplicationContext ctx = new ClassPathXmlApplicatiionContext("applicationContext.xml");
User user = ctx.getBean("user"); //我们直接用最底层的接口 BeanFactory 获取xml信息 BeanFactory bf = new XmlBeanFactory(new ClassPathResource("applicationContext.xml"));
User user = bf.getBean("user");

说明:

  1. beanFactory 底层获取xml文件信息的实现类 XmlBeanFactory 需要传入一个 ClassPathResource 对象。这个对象的父接口就是InputStreamSource 他就是提供了一个获取输入流的方法
    InputStream getInputStream() throws IOException;

    而ClassPathResource 对这个方法的实现就是通过类或者类加载器实现的

    if (this.clazz != null) {
    is = this.clazz.getResourceAsStream(this.path);
    }
    else if (this.classLoader != null) {
    is = this.classLoader.getResourceAsStream(this.path);
    }

    获取了输入流那就自然可以获取文件的内容了。

  2. Spring获取xml内容后通过XmlBeanDefinitionReader解析配置文件的内容封装成BeanDefinition方便后续使用。
    //前边说了是XmlBeanFactory具体实现获取xml信息的功能
    1. public class XmlBeanFactory extends DefaultListableBeanFactory {
    //xmlBeanFactory 中直接实例化 xmlBeanDefinitionReader
    private final XmlBeanDefinitionReader reader = new XmlBeanDefinitionReader(this);
    .....
    public XmlBeanFactory(Resource resource, BeanFactory parentBeanFactory) throws BeansException {
    super(parentBeanFactory);
    this.reader.loadBeanDefinitions(resource);
    }
    }
    2. public class XmlBeandefinitionReader extends AbstractBeanDefinitionReader{
    public int loadBeanDefinitions(EncodedResource encodedResource) throws BeanDefinitionStoreException {
    ......
    try (
    //获取输入流
    InputStream inputStream = encodedResource.getResource().getInputStream()) {
    //xml 解析工具类
    InputSource inputSource = new InputSource(inputStream);
    if (encodedResource.getEncoding() != null) {
    inputSource.setEncoding(encodedResource.getEncoding());
    }
    //开始具体解析
    return doLoadBeanDefinitions(inputSource, encodedResource.getResource());
    }
    ......
    }; protected int doLoadBeanDefinitions(InputSource inputSource, Resource resource)
    throws BeanDefinitionStoreException {
    try {
    // xml 解析为 document
    Document doc = doLoadDocument(inputSource, resource);
    // document 转 beanDefinition
    int count = registerBeanDefinitions(doc, resource);
    if (logger.isDebugEnabled()) {
    logger.debug("Loaded " + count + " bean definitions from " + resource);
    }
    return count;
    }
    }
    }
    3. public class DefaultBeanDefinitionDocumentReader implements BeanDefinitionDocumentReader {
    protected void doRegisterBeanDefinitions(Element root){
    ...
    preProcessXml(root);
    parseBeanDefinitions(root,this,dalegate);
    postProcessXml(root);
    }; protected void parseBeanDefinitions(Element root,BeanDefinitionParseDelegate delegate){
    if (delegate.isDefaultNamespace(root)) {
    NodeList nl = root.getChildNodes();
    for (int i = 0; i < nl.getLength(); i++) {
    Node node = nl.item(i);
    if (node instanceof Element) {
    Element ele = (Element) node;
    if (delegate.isDefaultNamespace(ele)) {
    //解析基本标签
    parseDefaultElement(ele, delegate);
    }
    else {
    //解析自定义标签
    delegate.parseCustomElement(ele);
    }
    }
    }
    }
    } }

    解释说明:

    1. 通过 XmlBeanDefinitionReader 的 loadBeanDefinitions 方法得到输入流和xml解析工具类在 doLoadBeanDefinitions方法中把输入流也就是获得的xml文件信息转化为 Document 再通过 registerBeanDefinitions 方法封装成 beanDefinition
    2. 封装beanDefinition是在 DefaultBeanDefinitionDocumentReader 类中的doRegisterBeanDefinitions . parseBeanDefinitions方法做了具体功能的实现,也就是解析文件中的标签并和beanDefinition的属性做映射。例如: 根标签(beans、prefile等) 子标签 (基本标签 bean、import、alias等,自定义标签 aop、mvc:annotation-driven、tx:annotation-driven、context:等)
    3. 用 BeanDefinitionParserDelegate 把解析标签得到的值映射成beanDefinition方便后续使用。

最后

感谢您的阅读,有什么意见和问题欢迎评论区留言!书写不易!

觉得文章对你有帮助记得给我点个赞,欢迎大家关注和转发文章!

最新文章

  1. 关于 CSS 反射倒影的研究思考
  2. CocoaPods报错:The dependency `Alamofire ` is not used in any concrete target
  3. iOS应用数据存储2-SQLite3数据库
  4. PL-SQL 存储函数和存储过程
  5. 【sinatra】设置默认的端口
  6. [mondrian] 分析一个简单的schema文件
  7. Linux tar指令
  8. Zookeeper ZAB 协议分析
  9. Leetcode Scratching Recorder
  10. react 阻止事件冒泡
  11. react native 5.54 出ios版本遇到的坑(应该是在xcode10下才会有的吧)记录。。。。。。 据说5.7已经修复了
  12. 简析服务端通过geotools导入SHP至PG的方法
  13. Linux kernel Programming - Concurrency and Race Conditions
  14. mysql逗逼的.frm文件恢复数据库
  15. 浏览器输入URL后发生了什么
  16. ThreeJS实现波纹粒子效果
  17. PowerDesigner15.1使用技巧总结
  18. 使用memset初始化C++自定义类型
  19. Red Hat 6.5 Samba服务器的搭建(登录访问)
  20. 1.linux环境配置

热门文章

  1. mysql:刚刚知道的冷知识(一)
  2. cs_play
  3. JSTL标签报错-http://java.sun.com/jsp/jstl/core
  4. Redis详解(二)——
  5. Vue CSS模拟菜单点击变色
  6. AI:用软件逻辑做硬件爆款
  7. Dubbo No provider问题排查思路
  8. 硕盟USB3.0 转RJ45千兆网卡 TYPE A USB3.0 TO RJ45
  9. PyCharm--帮助文档
  10. POJ题目 1003Hangover(叠放纸牌)