详解Java解析XML的四种方法

Posted XDY向东源

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了详解Java解析XML的四种方法相关的知识,希望对你有一定的参考价值。

XML现在已经成为一种通用的数据交换格式,它的平台无关性,语言无关性,系统无关性,给数据集成与交互带来了极大的方便。对于XML本身的语法知识与技术细节,需要阅读相关的技术文献,这里面包括的内容有DOM(Document Object Model),DTD(Document Type Definition),SAX(Simple API for XML),XSD(Xml Schema Definition),XSLT(Extensible Stylesheet Language Transformations),具体可参阅w3c官方网站文档http://www.w3.org获取更多信息。

XML在不同的语言里解析方式都是一样的,只不过实现的语法不同而已。基本的解析方式有两种,一种叫SAX,另一种叫DOM。SAX是基于事件流的解析,DOM是基于XML文档树结构的解析。假设我们XML的内容和结构如下:

ddviplinux 

30 

本文使用JAVA语言来实现DOM与SAX的XML文档生成与解析。 

首先定义一个操作XML文档的接口XmlDocument 它定义了XML文档的建立与解析的接口。

package com.alisoft.facepay.framework.bean; 

/** 

* 

* @author hongliang.dinghl 

* 定义XML文档建立与解析的接口 

*/ 

public interface XmlDocument  

/** 

* 建立XML文档 

* @param fileName 文件全路径名称 

*/ 

public void createXml(String fileName); 

/** 

* 解析XML文档 

* @param fileName 文件全路径名称 

*/ 

public void parserXml(String fileName); 

 

1.DOM生成和解析XML文档

为 XML 文档的已解析版本定义了一组接口。解析器读入整个文档,然后构建一个驻留内存的树结构,然后代码就可以使用 DOM 接口来操作这个树结构。优点:整个文档树在内存中,便于操作;支持删除、修改、重新排列等多种功能;缺点:将整个文档调入内存(包括无用的节点),浪费时间和空间;使用场合:一旦解析了文档还需多次访问这些数据;硬件资源充足(内存、CPU)。 

package com.alisoft.facepay.framework.bean; 

import java.io.FileInputStream; 

import java.io.FileNotFoundException; 

import java.io.FileOutputStream; 

import java.io.IOException; 

import java.io.InputStream; 

import java.io.PrintWriter; 

import javax.xml.parsers.DocumentBuilder; 

import javax.xml.parsers.DocumentBuilderFactory; 

import javax.xml.parsers.ParserConfigurationException; 

import javax.xml.transform.OutputKeys; 

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 org.w3c.dom.Document; 

import org.w3c.dom.Element; 

import org.w3c.dom.Node; 

import org.w3c.dom.NodeList; 

import org.xml.sax.SAXException; 

/** 

* 

* @author hongliang.dinghl 

* DOM生成与解析XML文档 

*/ 

public class DomDemo implements XmlDocument  

private Document document; 

private String fileName; 

public void init()  

try  

DocumentBuilderFactory factory = DocumentBuilderFactory 

.newInstance(); 

DocumentBuilder builder = factory.newDocumentBuilder(); 

this.document = builder.newDocument(); 

 catch (ParserConfigurationException e)  

System.out.println(e.getMessage()); 

 

 

public void createXml(String fileName)  

Element root = this.document.createElement("employees"); 

this.document.appendChild(root); 

Element employee = this.document.createElement("employee"); 

Element name = this.document.createElement("name"); 

name.appendChild(this.document.createTextNode("丁宏亮")); 

employee.appendChild(name); 

Element sex = this.document.createElement("sex"); 

sex.appendChild(this.document.createTextNode("m")); 

employee.appendChild(sex); 

Element age = this.document.createElement("age"); 

age.appendChild(this.document.createTextNode("30")); 

employee.appendChild(age); 

root.appendChild(employee); 

TransformerFactory tf = TransformerFactory.newInstance(); 

try  

Transformer transformer = tf.newTransformer(); 

DOMSource source = new DOMSource(document); 

transformer.setOutputProperty(OutputKeys.ENCODING, "gb2312"); 

transformer.setOutputProperty(OutputKeys.INDENT, "yes"); 

PrintWriter pw = new PrintWriter(new FileOutputStream(fileName)); 

StreamResult result = new StreamResult(pw); 

transformer.transform(source, result); 

System.out.println("生成XML文件成功!"); 

 catch (TransformerConfigurationException e)  

System.out.println(e.getMessage()); 

 catch (IllegalArgumentException e)  

System.out.println(e.getMessage()); 

 catch (FileNotFoundException e)  

System.out.println(e.getMessage()); 

 catch (TransformerException e)  

System.out.println(e.getMessage()); 

 

 

public void parserXml(String fileName)  

try  

DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance(); 

DocumentBuilder db = dbf.newDocumentBuilder(); 

Document document = db.parse(fileName); 

NodeList employees = document.getChildNodes(); 

for (int i = 0; i < employees.getLength(); i++)  

Node employee = employees.item(i); 

NodeList employeeInfo = employee.getChildNodes(); 

for (int j = 0; j < employeeInfo.getLength(); j++)  

Node node = employeeInfo.item(j); 

NodeList employeeMeta = node.getChildNodes(); 

for (int k = 0; k < employeeMeta.getLength(); k++)  

System.out.println(employeeMeta.item(k).getNodeName() 

+ ":" + employeeMeta.item(k).getTextContent()); 

 

 

 

System.out.println("解析完毕"); 

 catch (FileNotFoundException e)  

System.out.println(e.getMessage()); 

 catch (ParserConfigurationException e)  

System.out.println(e.getMessage()); 

 catch (SAXException e)  

System.out.println(e.getMessage()); 

 catch (IOException e)  

System.out.println(e.getMessage()); 

 

 

 

2.SAX生成和解析XML文档

为解决DOM的问题,出现了SAX。SAX ,事件驱动。当解析器发现元素开始、元素结束、文本、文档的开始或结束等时,发送事件,程序员编写响应这些事件的代码,保存数据。优点:不用事先调入整个文档,占用资源少;SAX解析器代码比DOM解析器代码小,适于Applet,下载。缺点:不是持久的;事件过后,若没保存数据,那么数据就丢了;无状态性;从事件中只能得到文本,但不知该文本属于哪个元素;使用场合:Applet;只需XML文档的少量内容,很少回头访问;机器内存少;

Java代码:

package com.alisoft.facepay.framework.bean;   

import java.io.FileInputStream;   

import java.io.FileNotFoundException;   

import java.io.IOException;   

import java.io.InputStream;   

import javax.xml.parsers.ParserConfigurationException;   

import javax.xml.parsers.SAXParser;   

import javax.xml.parsers.SAXParserFactory;   

import org.xml.sax.Attributes;   

import org.xml.sax.SAXException;   

import org.xml.sax.helpers.DefaultHandler;   

/**  

*   

* @author hongliang.dinghl  

* SAX文档解析  

*/  

public class SaxDemo implements XmlDocument    

public void createXml(String fileName)    

System.out.println(">");   

   

public void parserXml(String fileName)    

SAXParserFactory saxfac = SAXParserFactory.newInstance();   

try    

SAXParser saxparser = saxfac.newSAXParser();   

InputStream is = new FileInputStream(fileName);   

saxparser.parse(is, new MySAXHandler());   

 catch (ParserConfigurationException e)    

e.printStackTrace();   

 catch (SAXException e)    

e.printStackTrace();   

 catch (FileNotFoundException e)    

e.printStackTrace();   

 catch (IOException e)    

e.printStackTrace();   

   

   

   

class MySAXHandler extends DefaultHandler    

boolean hasAttribute = false;   

Attributes attributes = null;   

public void startDocument() throws SAXException    

System.out.println("文档开始打印了");   

   

public void endDocument() throws SAXException    

System.out.println("文档打印结束了");   

   

public void startElement(String uri, String localName, String qName,   

Attributes attributes) throws SAXException    

if (qName.equals("employees"))    

return;   

   

if (qName.equals("employee"))    

System.out.println(qName);   

   

if (attributes.getLength() > 0)    

this.attributes = attributes;   

this.hasAttribute = true;   

   

   

public void endElement(String uri, String localName, String qName)   

throws SAXException    

if (hasAttribute && (attributes != null))    

for (int i = 0; i < attributes.getLength(); i++)    

System.out.println(attributes.getQName(0)   

+ attributes.getValue(0));   

   

   

   

public void characters(char[] ch, int start, int length)   

throws SAXException    

System.out.println(new String(ch, start, length));   

   

  

package com.alisoft.facepay.framework.bean; 

import java.io.FileInputStream; 

import java.io.FileNotFoundException; 

import java.io.IOException; 

import java.io.InputStream; 

import javax.xml.parsers.ParserConfigurationException; 

import javax.xml.parsers.SAXParser; 

import javax.xml.parsers.SAXParserFactory; 

import org.xml.sax.Attributes; 

import org.xml.sax.SAXException; 

import org.xml.sax.helpers.DefaultHandler; 

/** 

* 

* @author hongliang.dinghl 

* SAX文档解析 

*/ 

public class SaxDemo implements XmlDocument  

public void createXml(String fileName)  

System.out.println(">"); 

 

public void parserXml(String fileName)  

SAXParserFactory saxfac = SAXParserFactory.newInstance(); 

try  

SAXParser saxparser = saxfac.newSAXParser(); 

InputStream is = new FileInputStream(fileName); 

saxparser.parse(is, new MySAXHandler()); 

 catch (ParserConfigurationException e)  

e.printStackTrace(); 

 catch (SAXException e)  

e.printStackTrace(); 

 catch (FileNotFoundException e)  

e.printStackTrace(); 

 catch (IOException e)  

e.printStackTrace(); 

 

 

 

class MySAXHandler extends DefaultHandler  

boolean hasAttribute = false; 

Attributes attributes = null; 

public void startDocument() throws SAXException  

System.out.println("文档开始打印了"); 

 

public void endDocument() throws SAXException  

System.out.println("文档打印结束了"); 

 

public void startElement(String uri, String localName, String qName, 

Attributes attributes) throws SAXException  

if (qName.equals("employees"))  

return; 

 

if (qName.equals("employee"))  

System.out.println(qName); 

 

if (attributes.getLength() > 0)  

this.attributes = attributes; 

this.hasAttribute = true; 

 

 

public void endElement(String uri, String localName, String qName) 

throws SAXException  

if (hasAttribute && (attributes != null))  

for (int i = 0; i < attributes.getLength(); i++)  

System.out.println(attributes.getQName(0) 

+ attributes.getValue(0)); 

 

 

 

public void characters(char[] ch, int start, int length) 

throws SAXException  

System.out.println(new String(ch, start, length)); 

 

 

3.DOM4J生成和解析XML文档

DOM4J 是一个非常非常优秀的Java XML API,具有性能优异、功能强大和极端易用使用的特点,同时它也是一个开放源代码的软件。如今你可以看到越来越多的 Java 软件都在使用 DOM4J 来读写 XML,特别值得一提的是连 Sun 的 JAXM 也在用 DOM4J。

Java代码:

package com.alisoft.facepay.framework.bean;   

import java.io.File;   

import java.io.FileWriter;   

import java.io.IOException;   

import java.io.Writer;   

import java.util.Iterator;   

import org.dom4j.Document;   

import org.dom4j.DocumentException;   

import org.dom4j.DocumentHelper;   

import org.dom4j.Element;   

import org.dom4j.io.SAXReader;   

import org.dom4j.io.XMLWriter;   

/**  

*   

* @author hongliang.dinghl  

* Dom4j 生成XML文档与解析XML文档  

*/  

public class Dom4jDemo implements XmlDocument    

public void createXml(String fileName)    

Document document = DocumentHelper.createDocument();   

Element employees=document.addElement("employees");   

Element employee=employees.addElement("employee");   

Element name= employee.addElement("name");   

name.setText("ddvip");   

Element sex=employee.addElement("sex");   

sex.setText("m");   

Element age=employee.addElement("age");   

age.setText("29");   

try    

Writer fileWriter=new FileWriter(fileName);   

XMLWriter xmlWriter=new XMLWriter(fileWriter);   

xmlWriter.write(document);   

xmlWriter.close();   

 catch (IOException e)    

System.out.println(e.getMessage());   

   

   

public void parserXml(String fileName)    

File inputXml=new File(fileName);   

SAXReader saxReader = new SAXReader();   

try    

Document document = saxReader.read(inputXml);   

Element employees=document.getRootElement();   

for(Iterator i = employees.elementIterator(); i.hasNext();)   

Element employee = (Element) i.next();   

for(Iterator j = employee.elementIterator(); j.hasNext();)   

Element node=(Element) j.next();   

System.out.println(node.getName()+":"+node.getText());   

   

   

 catch (DocumentException e)    

System.out.println(e.getMessage());   

   

System.out.println("dom4j parserXml");   

   

   

4.JDOM生成和解析XML  

为减少DOM、SAX的编码量,出现了JDOM;优点:20-80原则,极大减少了代码量。使用场合:要实现的功能简单,如解析、创建等,但在底层,JDOM还是使用SAX(最常用)、DOM、Xanan文档。

package com.alisoft.facepay.framework.bean;   

import java.io.FileNotFoundException;   

import java.io.FileOutputStream;   

import java.io.IOException;   

import java.util.List;   

import org.jdom.Document;   

import org.jdom.Element;   

import org.jdom.JDOMException;   

import org.jdom.input.SAXBuilder;   

import org.jdom.output.XMLOutputter;   

/**  

*   

* @author hongliang.dinghl  

* JDOM 生成与解析XML文档  

*   

*/  

public class JDomDemo implements XmlDocument    

public void createXml(String fileName)    

Document document;   

Element  root;   

root=new Element("employees");   

document=new Document(root);   

Element employee=new Element("employee");   

root.addContent(employee);   

Element name=new Element("name");   

name.setText("ddvip");   

employee.addContent(name);   

Element sex=new Element("sex");   

sex.setText("m");   

employee.addContent(sex);   

Element age=new Element("age");   

age.setText("23");   

employee.addContent(age);   

XMLOutputter XMLOut = new XMLOutputter();   

try    

XMLOut.output(document, new FileOutputStream(fileName));   

 catch (FileNotFoundException e)    

e.printStackTrace();   

 catch (IOException e)    

e.printStackTrace();   

   

   

public void parserXml(String fileName)    

SAXBuilder builder=new SAXBuilder(false);    

try    

Document document=builder.build(fileName);   

Element employees=document.getRootElement();    

List employeeList=employees.getChildren("employee");   

for(int i=0;iElement employee=(Element)employeeList.get(i);   

List employeeInfo=employee.getChildren();   

for(int j=0;jSystem.out.println(((Element)employeeInfo.get(j)).getName()+":"+((Element)employeeInfo.get(j)).getValue());   

   

   

 catch (JDOMException e)    

e.printStackTrace();   

 catch (IOException e)    

e.printStackTrace();   

    

   

   

XML解析——Java中XML的四种解析方式

转:

XML解析——Java中XML的四种解析方式

XML是一种通用的数据交换格式,它的平台无关性、语言无关性、系统无关性、给数据集成与交互带来了极大的方便。XML在不同的语言环境中解析方式都是一样的,只不过实现的语法不同而已。

  XML的解析方式分为四种:1、DOM解析;2、SAX解析;3、JDOM解析;4、DOM4J解析。其中前两种属于基础方法,是官方提供的平台无关的解析方式;后两种属于扩展方法,它们是在基础的方法上扩展出来的,只适用于java平台。

  针对以下XML文件,会对四种方式进行详细描述:

XML

一、DOM解析

  DOM的全称是Document Object Model,也即文档对象模型。在应用程序中,基于DOM的XML分析器将一个XML文档转换成一个对象模型的集合(通常称DOM树),应用程序正是通过对这个对象模型的操作,来实现对XML文档数据的操作。通过DOM接口,应用程序可以在任何时候访问XML文档中的任何一部分数据,因此,这种利用DOM接口的机制也被称作随机访问机制。

  DOM接口提供了一种通过分层对象模型来访问XML文档信息的方式,这些分层对象模型依据XML的文档结构形成了一棵节点树。无论XML文档中所描述的是什么类型的信息,即便是制表数据、项目列表或一个文档,利用DOM所生成的模型都是节点树的形式。也就是说,DOM强制使用树模型来访问XML文档中的信息。由于XML本质上就是一种分层结构,所以这种描述方法是相当有效的。

  DOM树所提供的随机访问方式给应用程序的开发带来了很大的灵活性,它可以任意地控制整个XML文档中的内容。然而,由于DOM分析器把整个XML文档转化成DOM树放在了内存中,因此,当文档比较大或者结构比较复杂时,对内存的需求就比较高。而且,对于结构复杂的树的遍历也是一项耗时的操作。所以,DOM分析器对机器性能的要求比较高,实现效率不十分理想。不过,由于DOM分析器所采用的树结构的思想与XML文档的结构相吻合,同时鉴于随机访问所带来的方便,因此,DOM分析器还是有很广泛的使用价值的。

    优点:

      1、形成了树结构,有助于更好的理解、掌握,且代码容易编写。

      2、解析过程中,树结构保存在内存中,方便修改。

    缺点:

      1、由于文件是一次性读取,所以对内存的耗费比较大。

      2、如果XML文件比较大,容易影响解析性能且可能会造成内存溢出。

  以下是解析代码:

DOM

二、SAX解析

  SAX的全称是Simple APIs for XML,也即XML简单应用程序接口。与DOM不同,SAX提供的访问模式是一种顺序模式,这是一种快速读写XML数据的方式。当使用SAX分析器对XML文档进行分析时,会触发一系列事件,并激活相应的事件处理函数,应用程序通过这些事件处理函数实现对XML文档的访问,因而SAX接口也被称作事件驱动接口。

    优点:

      1、采用事件驱动模式,对内存耗费比较小。

      2、适用于只处理XML文件中的数据时。

    缺点:

      1、编码比较麻烦。

      2、很难同时访问XML文件中的多处不同数据。

  以下是解析代码:

SAX

 三、JDOM解析

    特征:

      1、仅使用具体类,而不使用接口。

      2、API大量使用了Collections类。

  以下是解析代码:

JDOM

4、DOM4J解析

     特征:

      1、JDOM的一种智能分支,它合并了许多超出基本XML文档表示的功能。

      2、它使用接口和抽象基本类方法。

      3、具有性能优异、灵活性好、功能强大和极端易用的特点。

      4、是一个开放源码的文件

  以下是解析代码:

复制代码
public class DOM4JTest {
    private static ArrayList<Book> bookList = new ArrayList<Book>();
    /**
     * @param args
     */
    public static void main(String[] args) {
        // 解析books.xml文件
        // 创建SAXReader的对象reader
        SAXReader reader = new SAXReader();
        try {
            // 通过reader对象的read方法加载books.xml文件,获取docuemnt对象。
            Document document = reader.read(new File("src/res/books.xml"));
            // 通过document对象获取根节点bookstore
            Element bookStore = document.getRootElement();
            // 通过element对象的elementIterator方法获取迭代器
            Iterator it = bookStore.elementIterator();
            // 遍历迭代器,获取根节点中的信息(书籍)
            while (it.hasNext()) {
                System.out.println("=====开始遍历某一本书=====");
                Element book = (Element) it.next();
                // 获取book的属性名以及 属性值
                List<Attribute> bookAttrs = book.attributes();
                for (Attribute attr : bookAttrs) {
                    System.out.println("属性名:" + attr.getName() + "--属性值:"
                            + attr.getValue());
                }
                Iterator itt = book.elementIterator();
                while (itt.hasNext()) {
                    Element bookChild = (Element) itt.next();
                    System.out.println("节点名:" + bookChild.getName() + "--节点值:" + bookChild.getStringValue());
                }
                System.out.println("=====结束遍历某一本书=====");
            }
        } catch (DocumentException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }
}
复制代码

Final:比较总结

  DOM4J性能最好,连Sun的JAXM也在用DOM4J。目前许多开源项目中大量采用DOM4J,例如大名鼎鼎的Hibernate也用DOM4J来读取XML配置文件。如果不考虑可移植性,那就采用DOM4J。
      JDOM和DOM在性能测试时表现不佳,在测试10M文档时内存溢出。在小文档情况下还值得考虑使用DOM和JDOM。虽然JDOM的开发者已经说明他们期望在正式发行版前专注性能问题,但是从性能观点来看,它确实没有值得推荐之处。另外,DOM仍是一个非常好的选择。DOM实现广泛应用于多种编程语言。它还是许多其它与XML相关的标准的基础,因为它正式获得W3C推荐(与基于非标准的Java模型相对),所以在某些类型的项目中可能也需要它(如在JavaScript中使用DOM)。
      SAX表现较好,这要依赖于它特定的解析方式-事件驱动。一个SAX检测即将到来的XML流,但并没有载入到内存(当然当XML流被读入时,会有部分文档暂时隐藏在内存中)。

以上是关于详解Java解析XML的四种方法的主要内容,如果未能解决你的问题,请参考以下文章

XML解析——Java中XML的四种解析方式

XML解析——Java中XML的四种解析方式

转载并整理JAVA解析或生成xml的四种方法

XML解析——Java中XML的四种解析方式 [转载]

XML的四种解析方式

XML--Java中的四种常见解析方式--jdom与dom4j