Exception in thread “main” java.lang.ClassCastException: com.sun.org.apache.xerces.internal.dom.Defe

  • Post author:
  • Post category:java

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

版权声明:本文为Georgezheng01原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。