如何使用SAX正确解析XML?

我从REST服务接收XML文档,该文档将使用SAX进行解析。 请参阅以下由XSD生成的示例。

设置解析器不是问题。 我的主要问题是startElement()endElement()方法等中的实际处理。我不明白如何提取我需要的项目并存储它们,因为它们有点“嵌套”。

ConnectionList可以出现一次或两次,并且可以包含任意数量的Connection元素,其中包含有关连接的详细信息。 基本上,我需要一个包含DateTransfersTime的所有连接的列表。 我是否必须为每个元素创建一个类?

据我所知,我不知何故需要做以下事情:如果解析器遇到…

  • ConnectionList :创建新的ConnectionList对象并将其放入ConnectionList列表中
  • Connection :创建一个新的Connection对象并将其放入Connections列表中
  • DateTransfersTime (仅当父级是Duration ):将节点值存储在当前的Connection对象中

我真的很感激任何帮助,提示,想法,摘要我是如何实现这一目标的。

谢谢 :-)

罗伯特

          0815     b3lcM_Yiyq7dqL9   
dZbgZfDH8j1hb1i 00d00:18:00 W5a47DRkc7XDZjhwq_s5Un.
GG56XN6zgiJF804mE_N4o t8mn634zjCZsRPyxj_e_-UYMH
D0MiUwW9nuuM_uykvawg2C07pwHL

我发现(到目前为止)使用SAX解析XML的最好方法是在相关的回调中使用堆栈和条件语句。 这是一篇描述它的文章 ,以及我对它的总结:

基本前提是,在解析文档时,您可以创建对象来存储已分析的数据,随时将它们推送到堆栈中,查看堆栈顶部以将数据添加到当前元素,并在每个元素的末尾元素将其从堆栈中弹出并将其存储在父级中。

结果是,您首先解析元素树的深度,然后在每个分支的末尾将其回滚到父级,直到您留下包含所有已解析数据的单个对象(例如您的ConnectionList)为止要使用的。 从本质上讲,您最终会得到一系列镜像原始XML结构的对象

这意味着您需要一些可以将数据存储在与XML相同的结构中的数据对象。 复杂元素通常会成为类,而简单元素通常是类中的属性。 根元素通常由某种列表表示。

首先,创建一个堆栈对象,以便在解析数据时保存数据。

然后,在每个元素的开头,使用localName.equals()方法确定它的类型,创建相应类的实例,并将其推入堆栈。 如果元素是一个简单元素,您可能会将其作为表示父元素的类中的属性进行建模,并且您将需要一系列标志来告诉解析器是否遇到这样的元素以及它是什么元素因此它可以在characters()方法中处理。

使用characters()方法读取实际数据,并再次使用条件逻辑根据标志的值确定如何处理数据。 基本上,您可以查看堆栈的顶部并使用适当的方法将数据写入对象,并在必要时从文本进行转换。

在每个元素的末尾,弹出堆栈的顶部并再次使用localName.equals()来确定如何将它存储在对象之前(例如,需要调用哪个setter方法)

当您到达文档的末尾时,您应该已捕获文档中的所有数据。

您的SAX事件处理程序应充当状态机。 你的结构很深,所以状态机会有点复杂; 但这是基本方法:

所有变量都是成员变量。

遇到startElement事件时,实例化表示该元素的对象,然后将该对象放在堆栈上(或设置一个标志,指示您正在使用的值)。

遇到文本事件时,请阅读文本并根据您在上一步中设置的标志设置适当的值。

遇到endElement事件时,将当前对象拉出堆栈并调用现在位于堆栈顶部的对象上的setter。

当你耗尽文档时,你应该只有一个对象留在堆栈上,代表你读过的所有内容。

如果它是一个相当小的xml文档,并且内存/吞吐量限制对于内存解决方案来说并不过分,那么您可以使用JAXB。 您可以从XSD生成所需的类,只需将xml解组为java对象即可。 如果你必须使用流解析器,那么考虑使用StAX,我通常会发现这更直观。

一般来说,你有几个选择:

  1. 使用自定义对象将XML映射到,这些对象将封装更多对象,就像XML元素嵌套一样。
  2. 进行通用解析并通过相关元素遍历DOM。

据我所知,有一些工具,比如JAXB,它们将根据XSD生成你的类,但它们有时会带来生成代码的价格。

如果你使用选项1并“自己动手”,则需要提供进出XML和最有可能的字符串的解组和编组方法。 就像是:

       // pseudo-code! //In Foo.java unmarshal( Element element ) { unmarshalBar( element ); unmarshalThing( element ); } unmarshalBar( Element element ) { //...verify that the element is bar bar = new Bar(); bar.unmarshal( element ); } //In Bar.java unmarshal( Element element ) { unmarshalBaz( element ); } 

希望这可以帮助。

我通常将对象放在堆栈上,并在解析XML文件时推送/弹出它们(如果对象是嵌套的,则特别有用,但这不是你的情况)。

如果您想要一种更简单的方法,则需要指向当前ConnectionList和当前Connection的指针。 由于您已经知道文件的结构,因此这比使用基于堆栈的解析器更容易。

SAX解析器有点像通过一个小间谍孔看大图。

回调将一次为您呈现一个XML结构。 它不会给你任何线索,因为你在文档中的位置只显示了一个数据。 元素名称,属性名称/值或文本内容。

您的程序需要跟踪您在文档中的位置。 如果您正在动态解析一个简单的堆栈结构 – 当您获得“beginelement”时,您将名称推送到堆栈上,然后将您的堆栈弹出“endelement”。

如果你发现自己正在构建一个树结构,我会切换到一个DOM解析器,因为你写的任何东西都会像XERCES这样苍白而多变的阴影。