解析XML时DTD的处理--解析时忽略,生成XML文件时加上。-- XML进阶

一、解析XML文件时,遇到DTD的定义要怎么办?

 

    下面这段代码是ibatis配置文件,用w3c Dom(其实任意一种解析方式都是这样)都会从http://www.ibatis.com/dtd/sql-map-config-2.dtd 这个地址找对应的DTD文件进行校验,如果因特网上这个地址不存在的话,就会报 java.net.ConnectException: Connection timed out: connect  。

 

 

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE sqlMapConfig PUBLIC "-//iBATIS.com//DTD SQL Map Config 2.0//EN" "http://www.ibatis.com/dtd/sql-map-config-2.dtd">
<sqlMapConfig>
<settings useStatementNamespaces="true"/>
<!--防止空SqlMap报错 by zhangbo start -->
<sqlMap resource="ibatis/project/project-blank.xml"/>

<sqlMap resource="ibatis/project/user-mapping.xml"/>
</sqlMapConfig>

 

   遇到这种问题可以有两种解析方式,第一把DTD下载的本地,指定从本地加载DTD文件;另一种,干脆忽略DTD文件。

这两种方式都是以重新构建实体的解析器(EntityResolver)为基础解决方案的,重新构建实体解析器而不是使用默认的实体解析器,因为默认的解析器会根据实际的url进行读取,下面是一段如何来解析上面提到的配置文件,两种方式都有标注

 

 

public void addToIbatisTotleConfig() {
		try {
			DocumentBuilderFactory factory = DocumentBuilderFactory
					.newInstance();
			
			factory.setIgnoringElementContentWhitespace(true);
			DocumentBuilder db = factory.newDocumentBuilder();			
			db.setEntityResolver(new EntityResolver(){
			public InputSource resolveEntity(String publicId, String systemId) throws SAXException, IOException {				 
				           if(publicId.equals("-//iBATIS.com//DTD SQL Map Config 2.0//EN")){
				        	  //这是第一种方式,即忽略DTD,新建一个默认的空XML来代替
				        	   return new InputSource(new ByteArrayInputStream("<?xml version='1.0' encoding='GBK'?>".getBytes()));
				        	   //这是第二种方式,即指定本地的DTD文件,当然,这种方式需要把XML文件下载到本地,放入对应的目录中
				        	  // InputStream dtd_stream = this.getClass().getResourceAsStream("..//dtd/sql-map-config-2.dtd");
				        	  //    return new InputSource(dtd_stream);

				           }				          
					return null;
				}});
			String ibatisTotalConfigFileName = ctx.getIbatisTotalConfigFile();
			File ibatisTotalConfigFile = new File(ibatisTotalConfigFileName);
			Document xmldoc = db.parse("file://localhost//"+ibatisTotalConfigFileName);
			Element root = xmldoc.getDocumentElement();
			} catch (Exception e) {
			e.printStackTrace();
		}
	}

 

 

二、使用W3CDom编辑完上面的XML文件,把document转换为XML文件的时候,DTD定义会丢失,如果要加上,要怎么办?

       加DTD定义时,要在转换器中操作,重点使用这个两句代码:

   transformer.setOutputProperty(javax.xml.transform.OutputKeys.DOCTYPE_PUBLIC, doc.getDoctype().getPublicId());  
	            transformer.setOutputProperty(javax.xml.transform.OutputKeys.DOCTYPE_SYSTEM, doc.getDoctype().getSystemId());

 

DTD定义的值在document对象中是有的,你可以直接输出,也可以改变。

下面是我的实际应用

 

import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;

import javax.xml.transform.Transformer;
import javax.xml.transform.TransformerConfigurationException;
import javax.xml.transform.TransformerException;
import javax.xml.transform.TransformerFactory;
import javax.xml.transform.dom.DOMSource;
import javax.xml.transform.stream.StreamResult;
import javax.xml.xpath.XPath;
import javax.xml.xpath.XPathConstants;
import javax.xml.xpath.XPathExpressionException;
import javax.xml.xpath.XPathFactory;

import org.w3c.dom.Document;
import org.w3c.dom.Node;	 



public static void saveXml(String fileName, Document doc) {// 将Document输出到文件   
	        TransformerFactory transFactory = TransformerFactory.newInstance();   
	        try {   
	            Transformer transformer = transFactory.newTransformer();   
	            transformer.setOutputProperty("indent", "yes");   
	  
	            DOMSource source = new DOMSource();   
	            source.setNode(doc);  
	            System.out.println(doc.getDoctype().getPublicId());
	             
	            transformer.setOutputProperty(javax.xml.transform.OutputKeys.DOCTYPE_PUBLIC, doc.getDoctype().getPublicId());  
	            transformer.setOutputProperty(javax.xml.transform.OutputKeys.DOCTYPE_SYSTEM, doc.getDoctype().getSystemId());
	            StreamResult result = new StreamResult();   
	            result.setOutputStream(new FileOutputStream(fileName));   
	  
	            transformer.transform(source, result);   
	        } catch (TransformerConfigurationException e) {   
	            e.printStackTrace();   
	        } catch (TransformerException e) {   
	            e.printStackTrace();   
	        } catch (FileNotFoundException e) {   
	            e.printStackTrace();   
	        }   
	    } 

 

三、用w3c dom 解析带有DTD定义的XML时,会给一些元素自动加些一默认属性,例如:下面Spring的代码:

 

<?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="no" default-dependency-check="none"
	default-lazy-init="false">
	<bean autowire="default"
		class="org.springframework.context.support.ClassPathXmlApplicationContext"
		dependency-check="default" id="galaxyBaseContext" lazy-init="true">
		<constructor-arg>
			<list>
				<value>spring/base-context.xml</value>
			</list>
		</constructor-arg>
	</bean>
	<bean autowire="default"
		class="org.springframework.context.support.ClassPathXmlApplicationContext"
		dependency-check="default" id="galaxyContext" lazy-init="true">
		<constructor-arg>
			<list>

 

default-autowire="no" default-dependency-check="none" default-lazy-init="false" 这三个属性是自动加的,其实我只是用W3c Dom解析后改变了一个bean元素的属性值,但是解析完,bean都会加前面三个属性,很麻烦,也可以用第一种方法解决,设置DTD忽略,忽略以后并不会影响第二点讲的解析完增加DTD。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值