【死磕 Spring】----- IOC 之 获取验证模型

在上篇博客【死磕Spring】----- IOC 之 加载 Bean 中提到,在核心逻辑方法 doLoadBeanDefinitions()中主要是做三件事情。

  1. 调用 getValidationModeForResource() 获取 xml 文件的验证模式
  2. 调用 loadDocument() 根据 xml 文件获取相应的 Document 实例。
  3. 调用 registerBeanDefinitions() 注册 Bean 实例。

这篇博客主要分析获取 xml 文件的验证模式。

XML 文件的验证模式保证了 XML 文件的正确性

DTD 与 XSD 的区别

DTD(Document Type Definition),即文档类型定义,为 XML 文件的验证机制,属于 XML 文件中组成的一部分。DTD 是一种保证 XML 文档格式正确的有效验证方式,它定义了相关 XML 文档的元素、属性、排列方式、元素的内容类型以及元素的层次结构。其实 DTD 就相当于 XML 中的 “词汇”和“语法”,我们可以通过比较 XML 文件和 DTD 文件 来看文档是否符合规范,元素和标签使用是否正确。

要在 Spring 中使用 DTD,需要在 Spring XML 文件头部声明:

<?xml version="1.0" encoding="UTF-8"?><!DOCTYPE beans PUBLIC  "-//SPRING//DTD BEAN//EN"  "http://www.springframework.org/dtd/spring-beans.dtd">

DTD 在一定的阶段推动了 XML 的发展,但是它本身存在着一些缺陷:

  1. 它没有使用 XML 格式,而是自己定义了一套格式,相对解析器的重用性较差;而且 DTD 的构建和访问没有标准的编程接口,因而解析器很难简单的解析 DTD 文档。
  2. DTD 对元素的类型限制较少;同时其他的约束力也叫弱。
  3. DTD 扩展能力较差。
  4. 基于正则表达式的 DTD 文档的描述能力有限。

针对 DTD 的缺陷,W3C 在 2001 年推出 XSD。XSD(XML Schemas Definition)即 XML Schema 语言。XML Schema 本身就是一个 XML文档,使用的是 XML 语法,因此可以很方便的解析 XSD 文档。相对于 DTD,XSD 具有如下优势:

  • XML Schema基于XML,没有专门的语法
  • XML Schema可以象其他XML文件一样解析和处理
  • XML Schema比DTD提供了更丰富的数据类型.
  • XML Schema提供可扩充的数据模型。
  • XML Schema支持综合命名空间
  • XML Schema支持属性组。

getValidationModeForResource() 分析

    protected int getValidationModeForResource(Resource resource) {        // 获取指定的验证模式        int validationModeToUse = getValidationMode();        // 如果手动指定,则直接返回        if (validationModeToUse != VALIDATION_AUTO) {            return validationModeToUse;        }        // 通过程序检测        int detectedMode = detectValidationMode(resource);        if (detectedMode != VALIDATION_AUTO) {            return detectedMode;        }        // 出现异常,返回 XSD        return VALIDATION_XSD;    }

如果指定了 XML 文件的的验证模式(调用 XmlBeanDefinitionReader.setValidating(booleanvalidating))则直接返回指定的验证模式,否则调用 detectValidationMode() 获取相应的验证模式,如下:

    protected int detectValidationMode(Resource resource) {        if (resource.isOpen()) {            throw new BeanDefinitionStoreException(                    "Passed-in Resource [" + resource + "] contains an open stream: " +                    "cannot determine validation mode automatically. Either pass in a Resource " +                    "that is able to create fresh streams, or explicitly specify the validationMode " +                    "on your XmlBeanDefinitionReader instance.");        }        InputStream inputStream;        try {            inputStream = resource.getInputStream();        }        catch (IOException ex) {            throw new BeanDefinitionStoreException(                    "Unable to determine validation mode for [" + resource + "]: cannot open InputStream. " +                    "Did you attempt to load directly from a SAX InputSource without specifying the " +                    "validationMode on your XmlBeanDefinitionReader instance?", ex);        }        try {          // 核心方法            return this.validationModeDetector.detectValidationMode(inputStream);        }        catch (IOException ex) {            throw new BeanDefinitionStoreException("Unable to determine validation mode for [" +                    resource + "]: an error occurred whilst reading from the InputStream.", ex);        }    }

前面一大堆的代码,核心在于 this.validationModeDetector.detectValidationMode(inputStream),validationModeDetector 定义为 XmlValidationModeDetector,所以验证模式的获取委托给 XmlValidationModeDetectordetectValidationMode() 方法。

    public int detectValidationMode(InputStream inputStream) throws IOException {        BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream));        try {            boolean isDtdValidated = false;            String content;            // 一行一行读取 xml 文件的内容            while ((content = reader.readLine()) != null) {                content = consumeCommentTokens(content);                if (this.inComment || !StringUtils.hasText(content)) {                    continue;                }                // 包含 DOCTYPE 为 DTD 模式                if (hasDoctype(content)) {                    isDtdValidated = true;                    break;                }                // 读取 < 开始符号,验证模式一定会在 < 符号之前                if (hasOpeningTag(content)) {                    // End of meaningful data...                    break;                }            }        // 为 true 返回 DTD,否则返回 XSD            return (isDtdValidated ? VALIDATION_DTD : VALIDATION_XSD);        }        catch (CharConversionException ex) {            // 出现异常,为 XSD            return VALIDATION_AUTO;        }        finally {            reader.close();        }    }

从代码中看,主要是通过读取 XML 文件的内容,判断内容中是否包含有 DOCTYPE ,如果是 则为 DTD,否则为 XSD,当然只会读取到 第一个 "<" 处,因为 验证模式一定会在第一个 “<” 之前。如果当中出现了 CharConversionException 异常,则为 XSD模式。

好了,XML 文件的验证模式分析完毕,下篇分析 doLoadBeanDefinitions() 的第二个步骤:获取 Document 实例。

原文发布于微信公众号 - Java技术驿站(chenssy89)

原文发表时间:2018-09-10

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏项勇

笔记44 | 数据存储之File存储最简单地实现

17340
来自专栏SpringBoot

java文件夹复制到指定目录

36410
来自专栏Script Boy (CN-SIMO)

文件随机读写专用类——RandomAccessFile

 RandomAccessFile类可以随机读取文件,但是在测试中并不好用; File类可以测试文件存不存在,不存在可以创建文件; FileWriter类可...

24600
来自专栏容器云生态

Golang读写文件操作

最近在使用Golang进行文件读写的过程中,遇到几个细节问题导致程序写入数据时有一定脏数据的残留,最后发现是使用os.OpenFile在进行文件操作的时候没有使...

96970
来自专栏Golang语言社区

Go语言中三种不同md5计算方式的性能比较

前言 本文主要介绍的是三种不同的 md5 计算方式,其实区别是读文件的不同,也就是磁盘 I/O, 所以也可以举一反三用在网络 I/O 上。下面来一起看看吧。 R...

32170
来自专栏影子

Java解析OFFICE(word,excel,powerpoint)以及PDF的实现方案及开发中的点滴分享

663160
来自专栏Golang语言社区

Go语言中三种不同md5计算方式的性能比较

前言 本文主要介绍的是三种不同的 md5 计算方式,其实区别是读文件的不同,也就是磁盘 I/O, 所以也可以举一反三用在网络 I/O 上。下面来一起看看吧。 R...

524100
来自专栏函数式编程语言及工具

Scalaz(16)- Monad:依赖注入-Dependency Injection By Reader Monad

  在上一篇讨论里我们简单的介绍了一下Cake Pattern和Reader Monad是如何实现依赖注入的。主要还是从方法上示范了如何用Cake Patter...

20380
来自专栏流柯技术学院

关于lr调用jar在vuser中可以运行,但是controller中却报错的问题

如题,错误如下:javax.xml.parsers.FactoryConfigurationError: Provider org.apache.xerces....

12420
来自专栏技术/开源

TypeScript设计模式之单例、建造者、原型

看看用TypeScript怎样实现常见的设计模式,顺便复习一下。 学模式最重要的不是记UML,而是知道什么模式可以解决什么样的问题,在做项目时碰到问题可以想到...

23760

扫码关注云+社区

领取腾讯云代金券