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

博客围绕使用DOM解析XML文档时出现的ClassCastException异常展开。分析得知,因xml文档元素间的空格符被系统当作文本节点,遍历ChildNodes时将文本节点强制转换为Element类型出错。给出两种解决方法,一是修改XML文档拿掉回车,二是判断是否为元素节点。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值