在不少狀況下,咱們須要爲系統提供可配置化支持,簡單的作法能夠直接基於Spring的標準Bean來配置,但配置較爲複雜或者須要更多豐富控制的 時候,會顯得很是笨拙。通常的作法會用原生態的方式去解析定義好的xml文件,而後轉化爲配置對象,這種方式固然能夠解決全部問題,但實現起來比較繁瑣, 特別是是在配置很是複雜的時候,解析工做是一個不得不考慮的負擔。Spring提供了可擴展Schema的支持,這是一個不錯的折中方案,完成一個自定義 配置通常須要如下步驟:spring
設計配置屬性和JavaBean
編寫XSD文件
編寫NamespaceHandler和BeanDefinitionParser完成解析工做
編寫spring.handlers和spring.schemas串聯起全部部件
在Bean文件中應用
下面結合一個小例子來實戰以上過程app
1)設計配置屬性和JavaBeandom
首先固然得設計好配置項,並經過JavaBean來建模,本例中須要配置People實體,配置屬性name和age(id是默認須要的)ide
public class People {
private String id;
private String name;
private Integer age;
}
2)編寫XSD文件ui
爲上一步設計好的配置項編寫XSD文件,XSD是schema的定義文件,配置的輸入和解析輸出都是以XSD爲契約,本例中XSD以下:spa
<?xml version="1.0" encoding="UTF-8"?>
<xsd:schema
xmlns="http://blog.csdn.net/cutesource/schema/people"
xmlns:xsd="http://www.w3.org/2001/XMLSchema"
xmlns:beans="http://www.springframework.org/schema/beans"
targetNamespace="http://blog.csdn.net/cutesource/schema/people"
elementFormDefault="qualified"
attributeFormDefault="unqualified">
<xsd:import namespace="http://www.springframework.org/schema/beans" />
<xsd:element name="people">
<xsd:complexType>
<xsd:complexContent>
<xsd:extension base="beans:identifiedType">
<xsd:attribute name="name" type="xsd:string" />
<xsd:attribute name="age" type="xsd:int" />
</xsd:extension>
</xsd:complexContent>
</xsd:complexType>
</xsd:element>
</xsd:schema>
關於xsd:schema的各個屬性具體含義就不做過多解釋,能夠參見http://www.w3school.com.cn/schema/schema_schema.asp.net
<xsd:element name="people">對應着配置項節點的名稱,所以在應用中會用people做爲節點名來引用這個配置設計
<xsd:attribute name="name" type="xsd:string" />和<xsd:attribute name="age" type="xsd:int" />對應着配置項people的兩個屬性名,所以在應用中能夠配置name和age兩個屬性,分別是string和int類型orm
完成後需把xsd存放在classpath下,通常都放在META-INF目錄下(本例就放在這個目錄下)xml
3)編寫NamespaceHandler和BeanDefinitionParser完成解析工做
下面須要完成解析工做,會用到NamespaceHandler和BeanDefinitionParser這兩個概念。具體說來 NamespaceHandler會根據schema和節點名找到某個BeanDefinitionParser,而後由 BeanDefinitionParser完成具體的解析工做。所以須要分別完成NamespaceHandler和 BeanDefinitionParser的實現類,Spring提供了默認實現類NamespaceHandlerSupport和 AbstractSingleBeanDefinitionParser,簡單的方式就是去繼承這兩個類。本例就是採起這種方式:
import org.springframework.beans.factory.xml.NamespaceHandlerSupport;
public class MyNamespaceHandler extends NamespaceHandlerSupport {
public void init() {
registerBeanDefinitionParser("people", new PeopleBeanDefinitionParser());
}
}
其中registerBeanDefinitionParser("people", new PeopleBeanDefinitionParser());就是用來把節點名和解析類聯繫起來,在配置中引用people配置項時,就會用 PeopleBeanDefinitionParser來解析配置。PeopleBeanDefinitionParser就是本例中的解析類:
import org.springframework.beans.factory.support.BeanDefinitionBuilder;
import org.springframework.beans.factory.xml.AbstractSingleBeanDefinitionParser;
import org.springframework.util.StringUtils;
import org.w3c.dom.Element;
public class PeopleBeanDefinitionParser extends AbstractSingleBeanDefinitionParser {
protected Class getBeanClass(Element element) {
return People.class;
}
protected void doParse(Element element, BeanDefinitionBuilder bean) {
String name = element.getAttribute("name");
String age = element.getAttribute("age");
String id = element.getAttribute("id");
if (StringUtils.hasText(id)) {
bean.addPropertyValue("id", id);
}
if (StringUtils.hasText(name)) {
bean.addPropertyValue("name", name);
}
if (StringUtils.hasText(age)) {
bean.addPropertyValue("age", Integer.valueOf(age));
}
}
}
其中element.getAttribute就是用配置中取得屬性值,bean.addPropertyValue就是把屬性值放到bean中。
4)編寫spring.handlers和spring.schemas串聯起全部部件
上面幾個步驟走下來會發現開發好的handler與xsd還無法讓應用感知到,就這樣放上去是無法把前面作的工做歸入體系中的,spring提供了 spring.handlers和spring.schemas這兩個配置文件來完成這項工做,這兩個文件須要咱們本身編寫並放入META-INF文件夾 中,這兩個文件的地址必須是META-INF/spring.handlers和META-INF/spring.schemas,spring會默認去 載入它們,本例中spring.handlers以下所示:
http\://blog.csdn.net/cutesource/schema/people=study.schemaExt.MyNamespaceHandler
以上表示當使用到名爲"http://blog.csdn.net/cutesource/schema/people"的schema引用時,會經過study.schemaExt.MyNamespaceHandler來完成解析
spring.schemas以下所示:
http\://blog.csdn.net/cutesource/schema/people.xsd=META-INF/people.xsd
以上就是載入xsd文件
5)在Bean文件中應用
到此爲止一個簡單的自定義配置以完成,能夠在具體應用中使用了。使用方法很簡單,和配置一個普通的spring bean相似,只不過須要基於咱們自定義schema,本例中引用方式以下所示:
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:cutesource="http://blog.csdn.net/cutesource/schema/people"
xsi:schemaLocation="
http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
http://blog.csdn.net/cutesource/schema/people http://blog.csdn.net/cutesource/schema/people.xsd">
<cutesource:people id="cutesource" name="袁志俊" age="27"/>
</beans>
其中xmlns:cutesource="http://blog.csdn.net/cutesource/schema/people" 是用來指定自定義schema,xsi:schemaLocation用來指定xsd文件。<cutesource:people id="cutesource" name="zhijun.yuanzj" age="27"/>是一個具體的自定義配置使用實例。
最後就能夠在具體程序中使用基本的bean載入方式來載入咱們的自定義配置對象了,如:
ApplicationContext ctx = new ClassPathXmlApplicationContext("application.xml");
People p = (People)ctx.getBean("cutesource");
System.out.println(p.getId());
System.out.println(p.getName());
System.out.println(p.getAge());
會輸出:
cutesource
袁志俊
27
以上就是一個基於Spring可擴展Schema提供自定義配置支持實戰過程,一些複雜應用和技巧還有待挖掘