XmlBeanDefinitionReader.doLoadDocument() 方法中做了两件事情
- 一是调用 getValidationModeForResource() 获取 XML 的验证模式
- 二是调用 DocumentLoader.loadDocument() 获取 Document 对象
DocumentLoader 中只有一个方法 loadDocument()
,该方法接收五个参数:
- inputSource:加载 Document 的 Resource 源
- entityResolver:解析文件的解析器
- errorHandler:处理加载 Document 对象的过程的错误
- validationMode:验证模式
- namespaceAware:命名空间支持。如果要提供对 XML 名称空间的支持,则为true
该方法由 DocumentLoader 的默认实现类 DefaultDocumentLoader 实现
- 首先调用
createDocumentBuilderFactory()
创建 DocumentBuilderFactory - 再通过该 factory 创建 DocumentBuilder,最后解析 InputSource 返回 Document 对象
EntityResolver
通过 loadDocument()
获取 Document 对象时,有一个参数 entityResolver ,该参数是通过 getEntityResolver()
获取的
getEntityResolver()
返回指定的解析器,如果没有指定,则构造一个未指定的默认解析器
- 如果 ResourceLoader 不为 null,则根据指定的 ResourceLoader 创建一个 ResourceEntityResolver。
- 如果 ResourceLoader 为null,则创建 一个 DelegatingEntityResolver,该 Resolver 委托给默认的 BeansDtdResolver 和 PluggableSchemaResolver
- ResourceEntityResolver:继承自 EntityResolver ,通过 ResourceLoader 来解析实体的引用。
- DelegatingEntityResolver:EntityResolver 的实现,分别代理了 dtd 的 BeansDtdResolver 和 xml schemas 的 PluggableSchemaResolver。
- BeansDtdResolver : spring bean dtd 解析器。EntityResolver 的实现,用来从 classpath 或者 jar 文件加载 dtd。
- PluggableSchemaResolver:使用一系列 Map 文件将 schema url 解析到本地 classpath 资源
getEntityResolver()
返回 EntityResolver ,那这个 EntityResolver 到底是什么呢?
- 如果 SAX 应用程序需要实现自定义处理外部实体,则必须实现此接口并使用
setEntityResolver()
向 SAX 驱动器注册一个实例 - EntityResolver 的作用就是应用本身可以提供一个如何寻找验证文件的方法,即自定义实现
这两个参数的实际内容和具体的验证模式有关系。如下
- XSD 验证模式
- publicId:null
- systemId:http://www.springframework.org/schema/beans/spring-beans.xsd
- DTD 验证模式
- publicId:-//SPRING//DTD BEAN 2.0//EN
- systemId:http://www.springframework.org/dtd/spring-beans.dtd