Spring的自定义Schema是如何解析生效的
随着 Spring Boot 的自定义日渐流行,应用里的何解大部分配置都被隐藏了起来,我们仅需要关心真正的析生效业务内容, Controller,自定义 Service, Repository,拿起键盘就是何解一通业务代码的Coding,具体的析生效 Component Scan,View,自定义PlaceHolder ... 都可以抛在脑后。何解但其实这种零配置在 Java 应用开发中,析生效还真不太久。自定义 「由奢入俭难」,何解不少开发者都经历过 Spring XML 配置的析生效冗长,再回到这种配置确实不好受。自定义
但有些时候,何解由于配置的析生效内容在 Spring Boot这种零配置中并不能很好的实现,就需要我们仍使用 XML 的配置形式,然后再 ImportSource进来。或者一些项目受环境等影响,未使用Boot进行开发,此时也需要对配置有一定的了解。网站模板
那这次让我们往回看一些,看看在 XML 配置中,一些自定义的 Schema 内容,是如何融合到 Spring 中进行配置的。例如:
Spring data es
dubbo
还有许多这样的例子,我们不再一一罗列。但通过上述两个图,我们发现一个共同的特点:
都是通过schemaLocation将对应的schema引入 在对应的beans元素中增加更具体的自定义配置那这些自定义的配置,是在什么时候工作的呢?如何校验是否配置正确?
我们来看 Spring 中包含一个名为 spring.handlers的文件,所有的自定义扩展,都是通过这个文件生效的。spring 官方的aop, tx 也都是服务器租用这个原理。
这个文件在哪呢?
如上图所示,是META-INF/spring.handlers。文件内容也超级简单:
http\://www.springframework.org/schema/data/elasticsearch=org.springframework.data.elasticsearch.config.ElasticsearchNamespaceHandler
前面是各个schema前缀,后面是schema 对应的解析类。这个spring.handlers文件是什么时候加载的呢?
这个也是发生在解析自定义配置文件过程中,有一个resolve的过程,此时会将当前classLoader对应的所有包含spring.handlers文件加载过来。
我们再来看这个解析类,内容如下:
public class ElasticsearchNamespaceHandler extends NamespaceHandlerSupport { public ElasticsearchNamespaceHandler() { } public void init() { RepositoryConfigurationExtension extension = new ElasticsearchRepositoryConfigExtension(); RepositoryBeanDefinitionParser parser = new RepositoryBeanDefinitionParser(extension); this.registerBeanDefinitionParser("repositories", parser); this.registerBeanDefinitionParser("node-client", new NodeClientBeanDefinitionParser()); this.registerBeanDefinitionParser("transport-client", new TransportClientBeanDefinitionParser()); } }首先是继承自NamesapceHandlerSupport
然后在重写的init方法中注册了一系列的parser,每个parser对应一个字符串,就是我们在xml配置文件是使用的自定义内容,比如上面的es的配置
<elasticsearch:transport-client id="client" cluster-nodes="192.168.73.186:9300" cluster 源码库