Freemarker解析有DTD的XML带来的问题

Freemarker的prase函数可以解析xml文件,将xml文档树解析为map结构,这样我们就可以用xml作为数据源了。这在Freemarker对xml文件的增量更新中用处很大。
但是当xml带有dtd校验时,如下一段xml文件:
<?xml version="1.0" encoding="UTF-8"?>

<!DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN"
	"http://www.springframework.org/dtd/spring-beans.dtd">

<beans default-autowire="byName">
。。。

prase函数会去网络上下载DTD文件来对xml进行校验,如果拔掉网线,prase函数就会抛出异常,www.springframework.org未知域名。
因为涉及到网络通信,prase函数执行的会很慢。

去掉DTD校验解决方法:
可以到 http://www.docjar.com/html/api/freemarker/ext/dom/NodeModel.java.html
查看prase函数的源代码。Freemarker并没有解决方法,那么只有重写NodeModel这个类了。

解决示例:
1. 重写prase函数
public abstract class NodeModelWithoutDtd extends NodeModel {

	protected NodeModelWithoutDtd(Node node) {
		super(node);
	}

	public static NodeModel parse(InputSource is) throws SAXException,
			IOException, ParserConfigurationException {
		return parse(is, true, true);
	}

	public static NodeModel parse(InputSource is, boolean removeComments,
			boolean removePIs) throws SAXException, IOException,
			ParserConfigurationException {
		DocumentBuilder builder = getDocumentBuilderFactory()
				.newDocumentBuilder();
		
		//设置成空的DTD校验
		NullEntityResolver nullEntityResolver = new NullEntityResolver();
		builder.setEntityResolver(nullEntityResolver);
		Document doc = builder.parse(is);
		
		if (removeComments && removePIs) {
			NodeModel.simplify(doc);
		}
		else {
			if (removeComments)
				NodeModel.removeComments(doc);
			if (removePIs)
				NodeModel.removePIs(doc);
			
			NodeModel.mergeAdjacentText(doc);
		}
		
		return NodeModel.wrap(doc);
	}
}


2. 空DTD校验类
public class NullEntityResolver implements EntityResolver {

	static String emptyDtd = "";

    static ByteArrayInputStream byteIs = new ByteArrayInputStream(emptyDtd.getBytes());

    public InputSource resolveEntity(String publicId, String systemId) 
                              throws SAXException, IOException {
        return new InputSource(byteIs);
    }

}



然后用NodeModelWithoutDtd.prase("")就可以了~

你可能感兴趣的:(spring,xml,bean,freemarker,ext)