使用Apache Xerces解析XML文档

一、技术概述

在用Java解析XML时候,一般都使用现成XML解析器来完成,自己编码解析是一件很棘手的问题,对程序员要求很高,一般也没有专业厂商或者开源组织实现的好。

Java解析XML的原理图如下:

目前Java XML解析器有十多种之多,解析原理有二:使用基于事件的XML简单API(Simple API for XML)称为SAX和基于树和节点的文档对象模型(Document Object Module)称为DOM。Sun公司提供了Java API for XML Parsing(JAXP)接口来使用SAX和DOM,通过JAXP,我们可以使用任何与JAXP兼容的XML解析器。但并不是所有的Java XML解析工具都实现了SUN提供的接口。DOM4J就没有遵循SUN的JAXP规范,但非常优秀。

DOM 采用建立树形结构的方式访问 XML 文档,而 SAX 采用的事件模型。

DOM 是W3C组织推荐的处理XML的标准接口,DOM 解析器把 XML 文档转化为一个包含其内容的树,并可以对树进行遍历。用 DOM 解析模型的优点是编程容易,开发人员只需要调用建树的指令,然后利用navigation APIs访问所需的树节点来完成任务。可以很容易的添加和修改树中的元素。然而由于使用 DOM 解析器的时候需要处理整个 XML 文档,所以对性能和内存的要求比较高,尤其是遇到很大的 XML 文件的时候。由于它的遍历能力,DOM 解析器常用于 XML 文档需要频繁的改变的服务中。

SAX是由XML_DEV邮件列表的成员开发的,它不是某个官方机构的标准,也不由W3C组织或其他任何官方机构维护,但它是XML社区事实上的标准。虽然SAX只是“民间”标准,但是它在XML中的应用丝毫不比DOM少,几乎所有的XML解析器都支持它。SAX 解析器采用了基于事件的模型,它在解析 XML 文档的时候可以触发一系列的事件,当发现给定的tag的时候,它可以激活一个回调方法,告诉该方法制定的标签已经找到。SAX 对内存的要求通常会比较低,因为它让开发人员自己来决定所要处理的tag。特别是当开发人员只需要处理文档中所包含的部分数据时,SAX 这种扩展能力得到了更好的体现。但用 SAX 解析器的时候编码工作会比较困难,而且很难同时访问同一个文档中的多处不同数据。

DOM和SAX只是定义了一些接口,以及某些接口的默认实现,一个应用程序要想利用DOM或SAX访问XML文档,还需要一个实现了DOM或SAX的解析器(实现DOM和SAX中定义的接口,提供DOM和SAX定义的功能)。

Apache的Xerces是一个使用非常广泛的解析其,它提供了DOM和SAX的调用接口,并有多种语言的实现版本,要利用Xerces访问XML文档,只需要在应用程序中构造一个解析器实现类的对象。

这里面还存在一个问题,虽然使用的是标准的DOM和SAX接口,由于不同的XML解析器的实现类是不同的,如果使用另外一种解析器,仍然需要修改应用程序,只不过修改的代码量较小,需要更改XML解析器实现类的类名,然后重新编译、发布。

然而,对于兼容JAXP的XML解析器,我们可以选用JAXP API,这样在更换兼容JAXP解析器的时候,就不用对已发布的程序做任何的改动。

JAXP开发包由javax.xml包、org.w3c.dom包、org.xml.sax包及其子包组成。在javax.xml.parsers包中,定义了几个工厂类,用于加载DOM和SAX的实现类。

使用JAXP API解析XML的原理如下:

二、使用DOM解析XML

DOM的核心概念就是NODE(节点)。DOM在分析XML文档时,将组成XML文档的各个部分(元素,属性,文本,注释,处理指令等)映射成一个对象,这个对象就叫做节点。在内存中这些节点形成一棵树。整棵树就是一个节点,树中的每一个节点也是一棵树(子树)。DOM就是对这棵树的一个对象描述,我们通过访问树中的节点来存取XML文档的内容。

DOM定义了一个Node接口,用于表示文档树中一个节点。从这个接口派生出来更多的具体的接口。例如,表示整个文档的Document接口,表示XML文档中的一个元素的Element接口,表示元素属性的Attr接口,都是从Node接口派生而来的。在org.w3c.dom包中,表示XML文档各组成部分的接口的继承关系如图:

举个例子,引用java web开发大全,有改动:

students.xml 
---------------------------------------------------------------------- 
<?xml version="1.0" encoding="GB2312"?> 
<?xml-stylesheet type="text/xsl" href="students.xsl"?> 
<students> 
    <student sn="01"> 
        <name>张三</name> 
        <age>18</age> 
    </student> 
    <student sn="02"> 
        <name>李四</name> 
        <age>20</age> 
    </student> 
</students> 
----------------------------------------------------------------------

public class DOMStudentsInfo { 
    public static void main(String[] args) { 
        //获取生成 DOM 对象树的解析器 
        DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance(); 
        try { 
            //从 DocumentBuilderFactory获取 DocumentBuilder实例 
            DocumentBuilder db = dbf.newDocumentBuilder(); 
            File file = new File("students.xml"); 
            ////从 XML 文档获取 DOM 文档实例 
            Document doc = db.parse(file); 
            //获取某节点的集合 
            NodeList nl = doc.getElementsByTagName("student"); 
            int len = nl.getLength(); 
            for (int i = 0; i < len; i++) { 
                Element eltStu = (Element) nl.item(i); 
                Node eltName = eltStu.getElementsByTagName("name").item(0); 
                Node eltAge = eltStu.getElementsByTagName("age").item(0); 
                String name = eltName.getFirstChild().getNodeValue(); 
                String age = eltAge.getFirstChild().getNodeValue(); 
                System.out.print("姓名:"); 
                System.out.println(name); 
                System.out.print("年龄:"); 
                System.out.println(age); 
                System.out.println("----------------------------\n"); 
            } 
        } 
        catch (ParserConfigurationException e) { 
            e.printStackTrace(); 
        } 
        catch (SAXException e) { 
            e.printStackTrace(); 
        } 
        catch (IOException e) { 
            e.printStackTrace(); 
        } 
    } 
}

运行结果:

姓名:张三 
年龄:18 
---------------------------- 
姓名:李四 
年龄:20 
----------------------------

Process finished with exit code 0

关于Node接口的更多信息,可以参看JDK的API文档。

三、使用SAX解析XML

data_10k.xml 
---------------------------------------------------------------------- 
<?xml version="1.0" encoding="GB2312"?> 
<?xml-stylesheet type="text/xsl" href="students.xsl"?> 
<result> 
    <value> 
        <no>a1234</no> 
        <addr>四川省xx县xx镇xx路x段xx号</addr> 
    </value> 
    <value> 
        <no>b1234</no> 
        <addr>四川省xx市xx乡xx村xx组</addr> 
    </value> 
</result> 
----------------------------------------------------------------------

public class MyXMLReader2 extends DefaultHandler { 
    java.util.Stack tags = new java.util.Stack(); 
    public MyXMLReader2() { 
        super(); 
    } 
    public static void main(String args[]) { 
        long lasting = System.currentTimeMillis(); 
        try { 
            SAXParserFactory sf = SAXParserFactory.newInstance(); 
            SAXParser sp = sf.newSAXParser(); 
            MyXMLReader2 reader = new MyXMLReader2(); 
            sp.parse(new InputSource("data_10k.xml"), reader); 
        } catch (Exception e) { 
            e.printStackTrace(); 
        } 
        System.out.println("运行时间:" + (System.currentTimeMillis() - lasting) + " 毫秒"); 
    } 
    public void characters(char ch[], int start, int length) throws SAXException { 
        String tag = (String) tags.peek(); 
        if (tag.equals("no")) { 
            System.out.print("车牌号码:" + new String(ch, start, length)); 
        } 
        if (tag.equals("addr")) { 
            System.out.println(" 地址:" + new String(ch, start, length)); 
        } 
    } 
    public void startElement( 
            String uri, 
            String localName, 
            String qName, 
            Attributes attrs) { 
        tags.push(qName); 
    } 
}

运行结果:

车牌号码:a1234车牌号码: 
         地址:四川省xx县xx镇xx路x段xx号 
地址: 
    
地址: 
    
车牌号码:b1234车牌号码: 
         地址:四川省xx市xx乡xx村xx组 
地址: 
    
地址: 
运行时间:327 毫秒

Process finished with exit code 0

四、使用DOM4j解析xml

public class MyXMLReader4 { 
    public static void main(String arge[]) { 
        long lasting = System.currentTimeMillis(); 
        try { 
            File f = new File("data_10k.xml"); 
            SAXReader reader = new SAXReader(); 
            Document doc = reader.read(f); 
            Element root = doc.getRootElement(); 
            Element foo; 
            for (Iterator i = root.elementIterator("value"); i.hasNext();) { 
                foo = (Element) i.next(); 
                System.out.print("车牌号码:" + foo.elementText("no")); 
                System.out.println(" 车主地址:" + foo.elementText("addr")); 
            } 
        } catch (Exception e) { 
            e.printStackTrace(); 
        } 
        System.out.println("运行时间:" + (System.currentTimeMillis() - lasting) + " 毫秒"); 
    } 
}

五、使用JDOM解析xml

public class MyXMLReader3 { 
    public static void main(String arge[]) { 
        long lasting = System.currentTimeMillis(); 
        try { 
            SAXBuilder builder = new SAXBuilder(); 
            Document doc = builder.build(new File("data_10k.xml")); 
            Element foo = doc.getRootElement(); 
            List allChildren = foo.getChildren(); 
            for (int i = 0; i < allChildren.size(); i++) { 
                System.out.print("车牌号码:" + ((Element) allChildren.get(i)).getChild("no").getText()); 
                System.out.println(" 车主地址:" + ((Element) allChildren.get(i)).getChild("addr").getText()); 
            } 
        } catch (Exception e) { 
            e.printStackTrace(); 
        } 
        System.out.println("运行时间:" + (System.currentTimeMillis() - lasting) + " 毫秒"); 
    } 
}

六、使用JAXP解析XML

public class MyXMLReader1 { 
    public static void main(String arge[]) { 
        long lasting = System.currentTimeMillis(); 
        try { 
            File f = new File("data_10k.xml"); 
            DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance(); 
            DocumentBuilder builder = factory.newDocumentBuilder(); 
            Document doc = builder.parse(f); 
            NodeList nl = doc.getElementsByTagName("value"); 
            for (int i = 0; i < nl.getLength(); i++) { 
                System.out.print("车牌号码:" + doc.getElementsByTagName("no").item(i).getFirstChild().getNodeValue()); 
                System.out.println(" 车主地址:" + doc.getElementsByTagName("addr").item(i).getFirstChild().getNodeValue()); 
            } 
        } catch (Exception e) { 
            e.printStackTrace(); 
        } 
        System.out.println("运行时间:" + (System.currentTimeMillis() - lasting) + " 毫秒"); 
    } 
}

Java XML解析器的更多相关文章

  1. 使用java自带的xml解析器解析xml

    使用java自带的xml解析器解析xml,其实我不推荐,可以用Dom解析,或其他的方式,因为Java自带的解析器使用不但麻烦,且还有bug出现. 它要求,针对不同的xml(结构不同),必须写对应的ha ...

  2. tinyxml一个优秀的C++ XML解析器

    读取和设置xml配置文件是最常用的操作,试用了几个C++的XML解析器,个人感觉TinyXML是使用起来最舒服的,因为它的API接口和Java的十分类似,面向对象性很好. TinyXML是一个开源的解 ...

  3. TinyXML:一个优秀的C++ XML解析器

    //-------------------------------------------------------------------------------------------------- ...

  4. 转:TinyXM--优秀的C++ XML解析器

    读取和设置xml配置文件是最常用的操作,试用了几个C++的XML解析器,个人感觉TinyXML是使用起来最舒服的,因为它的API接口和Java的十分类似,面向对象性很好. TinyXML是一个开源的解 ...

  5. XML解析器(转)

    常见C/C++ XML解析器有tinyxml.XERCES.squashxml.xmlite.pugxml.libxml等等,这些解析器有些是支持多语言的,有些只是单纯C/C++的.如果你是第一次接触 ...

  6. XML 解析器

    所有现代浏览器都内建了供读取和操作 XML 的 XML 解析器.解析器把 XML 转换为 XML DOM 对象 - 可通过 JavaScript 操作的对象. 解析 XML 文档为DOM对象 方法一: ...

  7. TinyXML:一个优秀的C++ XML解析器[转]

    TinyXML:一个优秀的C++ XML解析器 读取和设置xml配置文件是最常用的操作,试用了几个C++的XML解析器,个人感觉TinyXML是使用起来最舒服的,因为它的API接口和Java的十分类似 ...

  8. XML概念定义以及如何定义xml文件编写约束条件java解析xml DTD XML Schema JAXP java xml解析 dom4j 解析 xpath dom sax

    本文主要涉及:xml概念描述,xml的约束文件,dtd,xsd文件的定义使用,如何在xml中引用xsd文件,如何使用java解析xml,解析xml方式dom sax,dom4j解析xml文件 XML来 ...

  9. JavaScript使用浏览器内置XML解析器解析DOM对象

    所有现代浏览器都内建了供读取和操作 XML 的 XML 解析器.解析器把 XML 转换为 XML DOM 对象 (可通过 JavaScript 操作的对象). 一.获取DOM对象 XMLHttpReq ...

随机推荐

  1. web.xml中常用元素的解读

    前言 针对一个项目而言,通常会有几类XML文件需要书写. web.xml spring-context.xml spring-mvc.xml other.xml ... 不管有多少配置文件,可以肯定的 ...

  2. stock 当天盘势

    看盘 (一)怎样看大盘当天的指数收阴.收阳 投资者每天都看大盘,但多数人看不准大盘.大盘到底当天是收阴,还是收阳?我通过很长时间的记录.验证,发现大盘在正常时间内,收阴.收阳是有规律的.我判断的准确率 ...

  3. oracle-创建表空间报错 提示ora-01119 ora-27040:无法创建文件

    create tablespace syx datafile 'D:\yangk\oraclespace\syx.ora' size 1000m; 低级错误,这是因为指定路径的路径问题,因为指定的路径 ...

  4. FusionCharts导出图表常见问题(FAQ)汇总---FusionCharts常见问题大全

    在前面几篇文章中,我们介绍了FusionCharts生成Flash图表常见问题FAQ以及使用中的一些常见报错及调试/解决方法.本文继续介绍FusionCharts导出图表时的一些常见问题(FAQ). ...

  5. opcode修改

    Smali: if-eqz opcode 38 if-nez opcode 39 SO: CMP R0,#0 00 28 CMP R0,#1 01 28

  6. mysql 字符集配置

    查看和设置MySQL数据库字符集作者:scorpio 2008-01-21 10:05:17 标签: 杂谈 Liunx下修改MySQL字符集:1.查找MySQL的cnf文件的位置find / -ina ...

  7. mysql 本地操作

    先把数据库传到根目录 再直接导入  没有50M限制root@b101 [/home/user/www]# mysql -uusername_007li -ppasswd -D dbname_li12d ...

  8. 高效判断奇偶性,利用位运算符&

    这种位运算判断奇偶性,在程序和数据库里面都是可以用的 public static bool isOdd(i) { return (i&1)!=0 } 最小奇数是:1   最小偶数是:0   所 ...

  9. SSH使用TCP Wrappers实现访问控制

    SSH使用TCP Wrappers实现访问控制主要配置文件/etc/hosts.allow/etc/hosts.deny===TCP Wrappers的访问控制原则首先检查 hosts.allow 文 ...

  10. POJ3268Dijkstra

    题意:给定n个点,m条边,求所有顶点中到顶点x的来回最短距离 分析:考虑到数据范围,选用Dijkstra,用Floyd会超时 #include <iostream> #include &l ...