Spring解析xml可以参考以上,可以指定自定义的schema,使用Jdk提供的xml API解析xml.
接下来Spring针对自己的schema,针对自己定义的xml元素,解析并注入到Spring的Bean中。
为了保持Spring的高可扩展性,用户可以在Spring的基础上最大限度的开放,这里采用了Schema Resolver,解析器采用最基本的Document Element.
这里给个例子并不是基本Spring的插件体系,不过原因相同。
package org.frame.base.xml.jdk.bk;
import java.io.IOException;
import java.io.StringReader;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.parsers.ParserConfigurationException;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.frame.base.xml.jdk.Contact;
import org.frame.base.xml.jdk.ContactName;
import org.frame.base.xml.jdk.MaySimpleSaxErrorHandler;
import org.frame.base.xml.jdk.MyPluggableSchemaResolver;
import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;
import org.xml.sax.EntityResolver;
import org.xml.sax.ErrorHandler;
import org.xml.sax.InputSource;
import org.xml.sax.SAXException;
/**
* Spring提供自定义schema,并提供自定义schema解析类, 并集成到Spring框架中
*
* @author ycl
* @version 1.0 2012-12-17 下午4:59:08
* @since 1.0
*
*/
public class TestSpringNamespaceHandler {
protected final static Log logger = LogFactory.getLog(TestSpringNamespaceHandler.class);
public static MyNamespaceHandlerResolver namespaceHandlerResolver = new DefaultMyNamespaceHandlerResolver();
public static void main(String[] args) {
// set jaxp debug
System.setProperty("jaxp.debug", "1");
DocumentBuilderFactory builderFactory = DocumentBuilderFactory
.newInstance();
// 设置使用命名空间
builderFactory.setNamespaceAware(true);
builderFactory.setValidating(true);
builderFactory.setIgnoringComments(true);
builderFactory.setAttribute(
"http://java.sun.com/xml/jaxp/properties/schemaLanguage",
"http://www.w3.org/2001/XMLSchema");
// 如果使用xsd,一船需要设置schemaLanguage的schema版本.
ErrorHandler errorHandler = new MaySimpleSaxErrorHandler(logger);
// 异常处理类
EntityResolver entityResolver = new MyPluggableSchemaResolver(
TestDocumentBuilderFactory.class.getClassLoader());
// 实体分解类
Document document = parse(builderFactory, getInputSource(),
entityResolver, errorHandler);
print(document);
printByHander(document);
}
private static void printByHander(Document document){
Map<String,Object> context = new HashMap<String,Object>();
Element root = document.getDocumentElement();
String namespaceUri = getNamespaceURI(root);
MyNamespaceHandler handler = namespaceHandlerResolver.resolve(namespaceUri);
handler.parse((Element) root,context);
//根元素解析
parserElement(root,context);
Contact contact = (Contact)context.get("contact");
System.out.println(contact);
}
private static void parserElement(Element root,Map<String,Object> context) {
NodeList nodes = root.getChildNodes();
if(nodes!=null&&nodes.getLength()>0){
for (int i = 0; i < nodes.getLength(); i++) {
Node node = nodes.item(i);
String namespaceUri = getNamespaceURI(node);
if(namespaceUri!=null){
MyNamespaceHandler handler = namespaceHandlerResolver.resolve(namespaceUri);
handler.parse((Element) node,context);
parserElement((Element)node,context);
}
}
}
}
private static void print(Document document) {
Element root = document.getDocumentElement();
List<ContactName> contactNameList = new ArrayList<ContactName>();
ContactName contactItem;
// 子元素列表
NodeList nodes = root.getChildNodes();
/**
* code this is so tied.
*/
for (int i = 0; i < nodes.getLength(); i++) {
Node node = nodes.item(i);
if (node instanceof Element) {
// a child element to process
Element child = (Element) node;
String width = child.getAttribute("width");
contactItem = new ContactName();
contactItem.setWidth(width);
NodeList itemSub = node.getChildNodes();
for (int j = 0; j < itemSub.getLength(); j++) {
Node itemSubNode = itemSub.item(j);
if (itemSubNode instanceof Element) {
if (((Element) itemSubNode).getTagName().equals("uic")) {
contactItem.setUid(itemSubNode.getTextContent());
} else if (((Element) itemSubNode).getTagName().equals(
"fullName")) {
contactItem.setFullName(itemSubNode
.getTextContent());
}
}
}
contactNameList.add(contactItem);
}
}
System.out.println(contactNameList);
}
/**
* 获取节点的namespace
* @param node
* @return
*/
public static String getNamespaceURI(Node node) {
return node.getNamespaceURI();
}
private static InputSource getInputSource() {
StringBuffer xml = new StringBuffer(
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>");
xml.append("<contact xmlns=\"http://www.ycl.com/schema/schema\" xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" xsi:schemaLocation=\"http://www.ycl.com/schema/schema http://www.ycl.com/schema/schema.xsd\" >");
xml.append("<!--ycl is good -->");
xml.append("<item width=\"10\">");
xml.append("<uic>1</uic>");
xml.append("<fullName>ycl1</fullName>");
xml.append("</item>");
xml.append("<item width=\"11\">");
xml.append("<uic>1 <![CDATA[06:00 Vesti<br>06:05 Jutarnji]]> 2</uic>");
xml.append("<fullName>ycl2</fullName>");
xml.append("</item>");
xml.append("</contact>");
InputSource is = new InputSource(new StringReader(xml.toString()));
return is;
}
private static Document parse(DocumentBuilderFactory builderFactory,
InputSource is, EntityResolver entityResolver,
ErrorHandler errorHandler) {
DocumentBuilder builder = null;
Document document = null;
try {
builder = builderFactory.newDocumentBuilder();
if (entityResolver != null) {
builder.setEntityResolver(entityResolver);
}
if (errorHandler != null) {
builder.setErrorHandler(errorHandler);
}
document = builder.parse(is);
} catch (ParserConfigurationException e) {
e.printStackTrace();
} catch (SAXException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return document;
}
}
测试类,主要是使用schema,并定义schema Resolver.
package org.frame.base.xml.jdk.bk;
public interface MyNamespaceHandlerResolver {
/**
* 对应schema 对应的 name handler
* @param namespaceUri
* @return
*/
MyNamespaceHandler resolve(String namespaceUri);
}
/*
* Copyright 2002-2009 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.frame.base.xml.jdk.bk;
import java.io.IOException;
import java.util.Map;
import java.util.Properties;
import java.util.concurrent.ConcurrentHashMap;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springframework.beans.BeanUtils;
import org.springframework.beans.FatalBeanException;
import org.springframework.beans.factory.xml.DefaultBeanDefinitionDocumentReader;
import org.springframework.beans.factory.xml.NamespaceHandler;
import org.springframework.beans.factory.xml.NamespaceHandlerResolver;
import org.springframework.core.io.support.PropertiesLoaderUtils;
import org.springframework.util.Assert;
import org.springframework.util.ClassUtils;
import org.springframework.util.CollectionUtils;
/**
* Default implementation of the {@link NamespaceHandlerResolver} interface.
* Resolves namespace URIs to implementation classes based on the mappings
* contained in mapping file.
*
* <p>By default, this implementation looks for the mapping file at
* <code>META-INF/spring.handlers</code>, but this can be changed using the
* {@link #DefaultNamespaceHandlerResolver(ClassLoader, String)} constructor.
*
* @author Rob Harrop
* @author Juergen Hoeller
* @since 2.0
* @see NamespaceHandler
* @see DefaultBeanDefinitionDocumentReader
*/
public class DefaultMyNamespaceHandlerResolver implements MyNamespaceHandlerResolver {
/**
* The location to look for the mapping files. Can be present in multiple JAR files.
*/
public static final String DEFAULT_HANDLER_MAPPINGS_LOCATION = "META-INF/myschema.handlers";
/** Logger available to subclasses */
protected final Log logger = LogFactory.getLog(getClass());
/** ClassLoader to use for NamespaceHandler classes */
private final ClassLoader classLoader;
/** Resource location to search for */
private final String handlerMappingsLocation;
/** Stores the mappings from namespace URI to NamespaceHandler class name / instance */
private volatile Map<String, Object> handlerMappings;
/**
* Create a new <code>DefaultNamespaceHandlerResolver</code> using the
* default mapping file location.
* <p>This constructor will result in the thread context ClassLoader being used
* to load resources.
* @see #DEFAULT_HANDLER_MAPPINGS_LOCATION
*/
public DefaultMyNamespaceHandlerResolver() {
this(null, DEFAULT_HANDLER_MAPPINGS_LOCATION);
}
/**
* Create a new <code>DefaultNamespaceHandlerResolver</code> using the
* default mapping file location.
* @param classLoader the {@link ClassLoader} instance used to load mapping resources
* (may be <code>null</code>, in which case the thread context ClassLoader will be used)
* @see #DEFAULT_HANDLER_MAPPINGS_LOCATION
*/
public DefaultMyNamespaceHandlerResolver(ClassLoader classLoader) {
this(classLoader, DEFAULT_HANDLER_MAPPINGS_LOCATION);
}
/**
* Create a new <code>DefaultNamespaceHandlerResolver</code> using the
* supplied mapping file location.
* @param classLoader the {@link ClassLoader} instance used to load mapping resources
* may be <code>null</code>, in which case the thread context ClassLoader will be used)
* @param handlerMappingsLocation the mapping file location
*/
public DefaultMyNamespaceHandlerResolver(ClassLoader classLoader, String handlerMappingsLocation) {
Assert.notNull(handlerMappingsLocation, "Handler mappings location must not be null");
this.classLoader = (classLoader != null ? classLoader : ClassUtils.getDefaultClassLoader());
this.handlerMappingsLocation = handlerMappingsLocation;
}
/**
* Locate the {@link NamespaceHandler} for the supplied namespace URI
* from the configured mappings.
* @param namespaceUri the relevant namespace URI
* @return the located {@link NamespaceHandler}, or <code>null</code> if none found
*/
public MyNamespaceHandler resolve(String namespaceUri) {
Map<String, Object> handlerMappings = getHandlerMappings();
Object handlerOrClassName = handlerMappings.get(namespaceUri);
if (handlerOrClassName == null) {
return null;
}
else if (handlerOrClassName instanceof MyNamespaceHandler) {
return (MyNamespaceHandler) handlerOrClassName;
}
else {
String className = (String) handlerOrClassName;
try {
Class<?> handlerClass = ClassUtils.forName(className, this.classLoader);
if (!MyNamespaceHandler.class.isAssignableFrom(handlerClass)) {
throw new FatalBeanException("Class [" + className + "] for namespace [" + namespaceUri +
"] does not implement the [" + MyNamespaceHandler.class.getName() + "] interface");
}
MyNamespaceHandler namespaceHandler = (MyNamespaceHandler) BeanUtils.instantiateClass(handlerClass);
namespaceHandler.init();
handlerMappings.put(namespaceUri, namespaceHandler);
return namespaceHandler;
}
catch (ClassNotFoundException ex) {
throw new FatalBeanException("NamespaceHandler class [" + className + "] for namespace [" +
namespaceUri + "] not found", ex);
}
catch (LinkageError err) {
throw new FatalBeanException("Invalid NamespaceHandler class [" + className + "] for namespace [" +
namespaceUri + "]: problem with handler class file or dependent class", err);
}
}
}
/**
* Load the specified NamespaceHandler mappings lazily.
*/
private Map<String, Object> getHandlerMappings() {
if (this.handlerMappings == null) {
synchronized (this) {
if (this.handlerMappings == null) {
try {
Properties mappings =
PropertiesLoaderUtils.loadAllProperties(this.handlerMappingsLocation, this.classLoader);
if (logger.isDebugEnabled()) {
logger.debug("Loaded NamespaceHandler mappings: " + mappings);
}
Map<String, Object> handlerMappings = new ConcurrentHashMap<String, Object>();
CollectionUtils.mergePropertiesIntoMap(mappings, handlerMappings);
this.handlerMappings = handlerMappings;
}
catch (IOException ex) {
throw new IllegalStateException(
"Unable to load NamespaceHandler mappings from location [" + this.handlerMappingsLocation + "]", ex);
}
}
}
}
return this.handlerMappings;
}
@Override
public String toString() {
return "NamespaceHandlerResolver using mappings " + getHandlerMappings();
}
}
请允许我的偷懒,赤裸裸的copy Spring 源码.
public interface MyNamespaceHandler {
public void init();
public ContactName parse(Element element,Map<String,Object> context);
}
package org.frame.base.xml.jdk.bk;
import java.io.IOException;
import java.util.HashMap;
import java.util.Map;
import java.util.Properties;
import java.util.concurrent.ConcurrentHashMap;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.frame.base.xml.jdk.ContactName;
import org.springframework.beans.BeanUtils;
import org.springframework.beans.FatalBeanException;
import org.springframework.core.io.support.PropertiesLoaderUtils;
import org.springframework.util.Assert;
import org.springframework.util.ClassUtils;
import org.springframework.util.CollectionUtils;
import org.w3c.dom.Element;
public class SchemaNamespaceHandler implements MyNamespaceHandler {
/**
* The location to look for the mapping files. Can be present in multiple JAR files.
*/
public static final String DEFAULT_HANDLER_MAPPINGS_LOCATION = "META-INF/schema/contact.handler";
/** Logger available to subclasses */
protected final Log logger = LogFactory.getLog(getClass());
/** ClassLoader to use for NamespaceHandler classes */
private final ClassLoader classLoader;
/** Resource location to search for */
private final String handlerMappingsLocation;
/** Stores the mappings from namespace URI to NamespaceHandler class name / instance */
private volatile Map<String, Object> handlerMappings;
public SchemaNamespaceHandler(){
this(null, DEFAULT_HANDLER_MAPPINGS_LOCATION);
}
public SchemaNamespaceHandler(ClassLoader classLoader) {
this(classLoader, DEFAULT_HANDLER_MAPPINGS_LOCATION);
}
public SchemaNamespaceHandler(ClassLoader classLoader, String handlerMappingsLocation) {
Assert.notNull(handlerMappingsLocation, "Handler mappings location must not be null");
this.classLoader = (classLoader != null ? classLoader : ClassUtils.getDefaultClassLoader());
this.handlerMappingsLocation = handlerMappingsLocation;
}
@Override
public void init() {
// TODO Auto-generated method stub
}
@Override
public ContactName parse(Element element,Map<String,Object> context) {
return resolve(element.getLocalName()).parse(element,context);
}
public MyNamespaceHandler resolve(String namespaceUri) {
Map<String, Object> handlerMappings = getHandlerMappings();
Object handlerOrClassName = handlerMappings.get(namespaceUri);
if (handlerOrClassName == null) {
return null;
}
else if (handlerOrClassName instanceof MyNamespaceHandler) {
return (MyNamespaceHandler) handlerOrClassName;
}
else {
String className = (String) handlerOrClassName;
try {
Class<?> handlerClass = ClassUtils.forName(className, this.classLoader);
if (!MyNamespaceHandler.class.isAssignableFrom(handlerClass)) {
throw new FatalBeanException("Class [" + className + "] for namespace [" + namespaceUri +
"] does not implement the [" + MyNamespaceHandler.class.getName() + "] interface");
}
MyNamespaceHandler namespaceHandler = (MyNamespaceHandler) BeanUtils.instantiateClass(handlerClass);
namespaceHandler.init();
handlerMappings.put(namespaceUri, namespaceHandler);
return namespaceHandler;
}
catch (ClassNotFoundException ex) {
throw new FatalBeanException("NamespaceHandler class [" + className + "] for namespace [" +
namespaceUri + "] not found", ex);
}
catch (LinkageError err) {
throw new FatalBeanException("Invalid NamespaceHandler class [" + className + "] for namespace [" +
namespaceUri + "]: problem with handler class file or dependent class", err);
}
}
}
/**
* Load the specified NamespaceHandler mappings lazily.
*/
private Map<String, Object> getHandlerMappings() {
if (this.handlerMappings == null) {
synchronized (this) {
if (this.handlerMappings == null) {
try {
Properties mappings =
PropertiesLoaderUtils.loadAllProperties(this.handlerMappingsLocation, this.classLoader);
if (logger.isDebugEnabled()) {
logger.debug("Loaded NamespaceHandler mappings: " + mappings);
}
Map<String, Object> handlerMappings = new ConcurrentHashMap<String, Object>();
CollectionUtils.mergePropertiesIntoMap(mappings, handlerMappings);
this.handlerMappings = handlerMappings;
}
catch (IOException ex) {
throw new IllegalStateException(
"Unable to load NamespaceHandler mappings from location [" + this.handlerMappingsLocation + "]", ex);
}
}
}
}
return this.handlerMappings;
}
@Override
public String toString() {
return "NamespaceHandlerResolver using mappings " + getHandlerMappings();
}
}
接下来看下我们的配置文件如下:
myschema.handlers
http\://www.ycl.com/schema/schema=org.frame.base.xml.jdk.bk.SchemaNamespaceHandler
contact.handler
contact=org.frame.base.xml.jdk.bk.ContactHandler
item=org.frame.base.xml.jdk.bk.ItemHandler
uic=org.frame.base.xml.jdk.bk.UicHandler
fullName=org.frame.base.xml.jdk.bk.FullNameHandler
我这里对xml中的每一个元素都定义了Handler
package org.frame.base.xml.jdk.bk;
import java.util.Map;
import org.frame.base.xml.jdk.Contact;
import org.frame.base.xml.jdk.ContactName;
import org.w3c.dom.Element;
/**
* 这里可以包含其他元素进行解析
*
* @author ycl
* @version 1.0 2012-12-18 上午9:32:12
* @since 1.0
*
*/
public class ContactHandler implements MyNamespaceHandler {
@Override
public void init() {
// TODO Auto-generated method stub
}
@Override
public ContactName parse(Element element,Map<String,Object> context) {
Contact contactName = new Contact();
context.put("contact", contactName);
return null;
}
}
package org.frame.base.xml.jdk.bk;
import java.util.Map;
import org.frame.base.xml.jdk.Contact;
import org.frame.base.xml.jdk.ContactName;
import org.frame.base.xml.jdk.Item;
import org.w3c.dom.Element;
public class ItemHandler implements MyNamespaceHandler {
@Override
public void init() {
// TODO Auto-generated method stub
}
@Override
public ContactName parse(Element element, Map<String, Object> context) {
Contact contactName = (Contact) context.get("contact");
String width = element.getAttribute("width");
Item item = new Item();
item.setWidth(width);
contactName.addItem(item);
context.put("curentItem", item);
return null;
}
}
package org.frame.base.xml.jdk.bk;
import java.util.Map;
import org.frame.base.xml.jdk.Contact;
import org.frame.base.xml.jdk.ContactName;
import org.frame.base.xml.jdk.Item;
import org.w3c.dom.Element;
public class UicHandler implements MyNamespaceHandler {
@Override
public void init() {
// TODO Auto-generated method stub
}
@Override
public ContactName parse(Element element, Map<String, Object> context) {
Item item = (Item)context.get("curentItem");
String uic = element.getTextContent();
item.setUic(uic);
return null;
}
}
package org.frame.base.xml.jdk.bk;
import java.util.Map;
import org.frame.base.xml.jdk.ContactName;
import org.frame.base.xml.jdk.Item;
import org.w3c.dom.Element;
public class FullNameHandler implements MyNamespaceHandler {
@Override
public void init() {
// TODO Auto-generated method stub
}
@Override
public ContactName parse(Element element, Map<String, Object> context) {
Item item = (Item)context.get("curentItem");
String fullName = element.getTextContent();
item.setFullName(fullName);
return null;
}
}
当然也,这里图个简单,解析整个xml,而在resolver中可以解析自己需要的xml元素到自己的Context中,context可以放任何中东西,这是一个上下文。
输出数据如下:
20121218-10:56:08 main org.springframework.beans.factory.xml.PluggableSchemaResolver Loading schema mappings from [META-INF/myschema.schemas]
20121218-10:56:08 main org.springframework.beans.factory.xml.PluggableSchemaResolver Loaded schema mappings: {http://www.ycl.com/schema/schema.xsd=org/frame/base/xml/jdk/schema.xsd}
20121218-10:56:08 main org.springframework.beans.factory.xml.PluggableSchemaResolver Found XML schema [http://www.ycl.com/schema/schema.xsd] in classpath: org/frame/base/xml/jdk/schema.xsd
[uid:1,fullName:ycl1,width:10, uid:1 06:00 Vesti<br>06:05 Jutarnji 2,fullName:ycl2,width:11]
20121218-10:56:08 main org.frame.base.xml.jdk.bk.DefaultMyNamespaceHandlerResolver Loaded NamespaceHandler mappings: {http://www.ycl.com/schema/schema=org.frame.base.xml.jdk.bk.SchemaNamespaceHandler}
20121218-10:56:08 main org.frame.base.xml.jdk.bk.SchemaNamespaceHandler Loaded NamespaceHandler mappings: {contact=org.frame.base.xml.jdk.bk.ContactHandler, uic=org.frame.base.xml.jdk.bk.UicHandler, item=org.frame.base.xml.jdk.bk.ItemHandler, fullName=org.frame.base.xml.jdk.bk.FullNameHandler}
[width:10,uic:1,fullName:ycl1,sex:null, width:11,uic:1 06:00 Vesti<br>06:05 Jutarnji 2,fullName:ycl2,sex:null]
这里可以输出自定义的handler,和自己想要的结果。
使用resolver 这种解析模式,你不需要关系xml结构,不需要关系xml到Java对象的关系。
你关心的只是xml元素,我需要解析xml中的元素到我的对象中,而xml元素和Java对象不存在必然的联系,这也是传说中的”解耦“吗.
我这里只是把每个元素都设计了Handler,如果使用这种解析模式,使用属性会让代码引人缩短,而且更易理解(怪不得bean这么多属性).
Spring对于Bean的设计也允许有子元素,那就是Properties,其实也是属性,只是放在子元素上看起来也更美观,逻辑更清晰,也更突显了Bean注入的强烈优势.