1、解决异常
Exception in thread “main” java.lang.ClassCastException: com.sun.org.apache.xerces.internal.dom.DeferredTextImpl cannot be cast to org.w3c.dom.Element at ParseXMLDemo02.main(ParseXMLDemo02.java:37)
2、场景
使用DOM解析XML文档时出现如上异常。
需求:使用DOM读取手机收藏信息
XML文档:
<?xml version="1.0" encoding="UTF-8"?>
<PhoneInfo>
<Brand name="华为">
<Type name="U8650"/>
</Brand>
<Brand name="苹果">
<Type name="iPhone4"/>
<Type name="iPhone5"/>
</Brand>
</PhoneInfo>
Java源代码:
import java.io.IOException;
import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.parsers.ParserConfigurationException;
import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;
import org.xml.sax.SAXException;
//演示:使用DOM从手机收藏信息中读取品牌和型号信息
public class ParseXMLDemo02 {
public static void main(String[] args) {
//得到DOM解析器的工厂实例
DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
try {
//从DOM工厂获得DOM解析器
DocumentBuilder builder = factory.newDocumentBuilder();
//解析XML文档,得到一个Document对象,即DOM树
Document document = builder.parse("src/收藏信息2.xml");
//得到所有Brand节点列表信息
NodeList brandList = document.getElementsByTagName("Brand");
//循环Brand信息
for(int i=0;i<brandList.getLength();i++){
//获取第i个Brand元素信息
Node brand = brandList.item(i);
//获取第i个Brand元素的name属性的值
Element element = (Element) brand;
String brandName = element.getAttribute("name");
//获取第i个Brand元素的所有子元素的name属性值
NodeList types = element.getChildNodes();
for(int j=0;j<types.getLength();j++){
Element typeElement = (Element) types.item(j);//Type节点
String typeName = typeElement.getAttribute("name");//获得手机型号
System.out.println("手机:"+brandName+typeName);
}
}
} catch (ParserConfigurationException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (SAXException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
3、问题分析即及解决
1)原因:在xml文档中元素之间使用了空格符,在getChildNodes()的时候系统把空格当成了文本节点,所以在遍历ChildNodes的时候将文本节点强制转换成Element类型就会出错。
2)解决方法:
方法一:
所谓元素间使用了空格键,指的就是使用回车或者空格,注意-包括回车。解决办法就是修改XML文档,拿掉回车。
修改后的XML文档:
<?xml version="1.0" encoding="UTF-8"?>
<PhoneInfo>
<Brand name="华为"><Type name="U8650"/></Brand>
<Brand name="苹果"><Type name="iPhone4"/><Type name="iPhone5"/></Brand>
</PhoneInfo>
方法二:
每次通过item()方法提取Node对象后判断node.getNodeType()==Node.ELEMENT_NODE,即判断是否是元素节点。原理:Element对象代表XML文档中的标签元素,继承自Node,也是Node最主要的子对象。
将36行至40行代码做如下修改:
for(int j=0;j<types.getLength();j++){
Node typeNode = types.item(j);
if(typeNode.getNodeType()==Node.ELEMENT_NODE){
String typeName = ((Element)typeNode.getAttribute("name");//获得手机型号
System.out.println("手机:"+brandName+typeName);
}
}
4、最终运行结果:
手机:华为U8650
手机:苹果iPhone4
手机:苹果iPhone5