【问题标题】:XML to Java ObjectsXML 到 Java 对象
【发布时间】:2012-09-12 13:15:08
【问题描述】:

我已经搜索了与此主题相关的所有问题,但我没有找到答案。我正在尝试将 xml 文档转换为 java 对象。在搜索了执行相同操作的选项后,我发现 jaxb 是执行相同操作的最佳选择。我参考了许多文章并提出了这个不起作用的解决方案。有人可以帮助我吗?

XML 国防部:

<Vectors>
    <Vector>
        <Id>1</Id>
        <Name>abc</Name>
         <Desc>abcdefg</Desc>
         <Type>body</Type>
         <Snippet>
            <![CDATA[<total>1234</total>]]>
         </Snippet>
    </Vector>
    <Vector>
         <Id>2</Id>
         <Name>def</Name>
         <Desc>abcdefghigj</Desc>
         <Type>body</Type>
         <Snippet>
            <![CDATA[<total>12345</total>]]>
         </Snippet>
     </Vector>
</Vectors>

我想创建 2 个 Vector 类的 java 对象。类向量如下所示

    package com.abc.TryXmltoJava;

    import java.util.List;
    import javax.xml.bind.annotation.*;

    public class Vector {
        int Id,
        String Name;
        String Desc;
        String Type;

        @XmlElement
        public String getName() {
             return Name;
        }
        public void setName(String name) {
             this.Name = name;
        }

        @XmlElement
        public String getDesc() {
            return Desc;
        }
        public void setDesc(String desc) {
            this.Desc = desc;
        }

        @XmlElement
        public String getType() {
            return Type;
        }
        public void setType(String type) {
            this.Type = type;
        }
}

现在我创建了 Vectors 类,它跟踪两个对象并使用列表实用程序创建 Vector 对象列表..

 package com.abc.TryXMLtoObject;

 import javax.xml.bind.annotation.*;

 @XmlRootElement(name="Vectors")
 public class Vectors {

 @XmlElement(name="Vector")
     Vector vector;


     public Vector getVector() {
        return vector;
     }

     public void setVector(Vector vector1) {
        this.vector = vector1;
     }
}

我将尝试使用这两个创建 java 对象列表的类来解组 xml 对象。解组类如下所示..

  package com.abc.TryXMLtoObject;

  import java.io.File;
  import javax.xml.bind.JAXBContext;
  import javax.xml.bind.JAXBException;
  import javax.xml.bind.Unmarshaller;

  public class JAXBExampleUnmarshal {

       public static void main(String[] args) {

         try {

          File file = new File("C:\\file.xml");
          JAXBContext jaxbContext = JAXBContext.newInstance(Vectors.class);

          Unmarshaller jaxbUnmarshaller = jaxbContext.createUnmarshaller();
          Vectors vectors = (Vectors) jaxbUnmarshaller.unmarshal(file);
          System.out.println(vectors);

         } catch (JAXBException e) {
          e.printStackTrace();
         }

   }
 }

运行此代码后..我收到以下错误:

javax.xml.bind.UnmarshalException: unexpected element (uri:"", local:"Vectors"). Expected elements are (none)
    at com.sun.xml.internal.bind.v2.runtime.unmarshaller.UnmarshallingContext.handleEvent(Unknown Source)
    at com.sun.xml.internal.bind.v2.runtime.unmarshaller.Loader.reportError(Unknown Source)
    at com.sun.xml.internal.bind.v2.runtime.unmarshaller.Loader.reportError(Unknown Source)
    at com.sun.xml.internal.bind.v2.runtime.unmarshaller.Loader.reportUnexpectedChildElement(Unknown Source)
    at com.sun.xml.internal.bind.v2.runtime.unmarshaller.UnmarshallingContext$DefaultRootLoader.childElement(Unknown Source)
    at com.sun.xml.internal.bind.v2.runtime.unmarshaller.UnmarshallingContext._startElement(Unknown Source)
    at com.sun.xml.internal.bind.v2.runtime.unmarshaller.UnmarshallingContext.startElement(Unknown Source)
    at com.sun.xml.internal.bind.v2.runtime.unmarshaller.SAXConnector.startElement(Unknown Source)
    at com.sun.org.apache.xerces.internal.parsers.AbstractSAXParser.startElement(Unknown Source)
    at com.sun.org.apache.xerces.internal.impl.XMLNSDocumentScannerImpl.scanStartElement(Unknown Source)
    at com.sun.org.apache.xerces.internal.impl.XMLNSDocumentScannerImpl$NSContentDriver.scanRootElementHook(Unknown Source)
    at com.sun.org.apache.xerces.internal.impl.XMLDocumentFragmentScannerImpl$FragmentContentDriver.next(Unknown Source)
    at com.sun.org.apache.xerces.internal.impl.XMLDocumentScannerImpl$PrologDriver.next(Unknown Source)
    at com.sun.org.apache.xerces.internal.impl.XMLDocumentScannerImpl.next(Unknown Source)
    at com.sun.org.apache.xerces.internal.impl.XMLNSDocumentScannerImpl.next(Unknown Source)
    at com.sun.org.apache.xerces.internal.impl.XMLDocumentFragmentScannerImpl.scanDocument(Unknown Source)
    at com.sun.org.apache.xerces.internal.parsers.XML11Configuration.parse(Unknown Source)
    at com.sun.org.apache.xerces.internal.parsers.XML11Configuration.parse(Unknown Source)
    at com.sun.org.apache.xerces.internal.parsers.XMLParser.parse(Unknown Source)
    at com.sun.org.apache.xerces.internal.parsers.AbstractSAXParser.parse(Unknown Source)
    at com.sun.org.apache.xerces.internal.jaxp.SAXParserImpl$JAXPSAXParser.parse(Unknown Source)
    at com.sun.xml.internal.bind.v2.runtime.unmarshaller.UnmarshallerImpl.unmarshal0(Unknown Source)
    at com.sun.xml.internal.bind.v2.runtime.unmarshaller.UnmarshallerImpl.unmarshal(Unknown Source)
    at javax.xml.bind.helpers.AbstractUnmarshallerImpl.unmarshal(Unknown Source)
    at javax.xml.bind.helpers.AbstractUnmarshallerImpl.unmarshal(Unknown Source)
    at javax.xml.bind.helpers.AbstractUnmarshallerImpl.unmarshal(Unknown Source)
    at javax.xml.bind.helpers.AbstractUnmarshallerImpl.unmarshal(Unknown Source)
    at com.abc.TryXMLtoObject.JAXBExampleUnmarshal.main(JAXBExampleUnmarshal.java:17)

【问题讨论】:

  • 尝试添加命名空间
  • 这可能是错字吗? (uri:“”,本地:“Vkectors”)。 “Vkectors”看起来不对。
  • 是的,这是一个错字。但我仍然遇到同样的错误,即本地:“Vectors”

标签: java xml jaxb


【解决方案1】:

一开始,你在Vectors你必须有List&lt;Vector&gt; vectorList,而不是一个单数Vector
此外,来自 stacktrace unexpected element (uri:"", local:"Vkectors") 的文本告诉我,在 xml 中的某处您有 Vkectors 元素,这是错误的。
最后,我强烈建议您将所有注释放在 get-methods 中,而不是变量,原因有两个:如果我没记错的话,有一些名称冲突,其次,如果您有来自 Hibernate 的代理或类似的东西,字段的注释将消失,但 get 方法的注释将保留。

【讨论】:

  • 哦.. 是的.. 会检查并让您知道它是否有效
  • +1 - 为避免在注释字段时出现名称冲突问题,应在类上指定@XmlAccessorType(XmlAccessType.FIELD)blog.bdoughan.com/2011/06/using-jaxbs-xmlaccessortype-to.html
  • 如果你有 field1@XmlElement 的注释字段,并且你有方法 getField1(),那么你会遇到名称冲突。不过,您可以使用访问器 getProperty1(){return field1;} 注释 field1,但这对我来说似乎很难看。
  • @popfalushi - 你看到我关于@XmlAccessorType(XmlAccessType.FIELD)的注释了吗?
  • 我很抱歉。我看到字段名称和访问器名称不同,并认为它证明了我的观点。现在在本地查了一下,发现你是对的。
【解决方案2】:

您的示例的以下版本对我有用:

向量

作为pointed out by popfalushiVectors 应该有一个List 属性,它可以保存Vector 的许多实例。

package com.abc.TryXmltoJava;

import java.util.List;
import javax.xml.bind.annotation.*;

@XmlRootElement(name="Vectors")
@XmlAccessorType(XmlAccessType.FIELD)
public class Vectors {

    @XmlElement(name="Vector")
    List<Vector> vector;

}

矢量

如果您希望在字段上指定 JAXB 注释,则应在类型级别指定 @XmlAccessorType(XmlAccessType.FIELD)(请参阅:http://blog.bdoughan.com/2011/06/using-jaxbs-xmlaccessortype-to.html)。此外,默认映射是 @XmlElement,因此您无需明确添加该注释(请参阅:http://blog.bdoughan.com/2012/07/jaxb-no-annotations-required.html)。

package com.abc.TryXmltoJava;

import javax.xml.bind.annotation.*;

@XmlAccessorType(XmlAccessType.FIELD)
public class Vector {
    int Id;
    String Name;
    String Desc;
    String Type;
}

JAXBExampleUnmarshal

我更新了您的演示代码以将对象写回 XML,以证明在解组操作期间对象已完全填充。

package com.abc.TryXmltoJava;

import java.io.File;
import javax.xml.bind.*;

public class JAXBExampleUnmarshal {

    public static void main(String[] args) {

        try {
            File file = new File("C:\\file.xml");
            JAXBContext jaxbContext = JAXBContext.newInstance(Vectors.class);

            Unmarshaller jaxbUnmarshaller = jaxbContext.createUnmarshaller();
            Vectors vectors = (Vectors) jaxbUnmarshaller.unmarshal(file);
            System.out.println(vectors);

            Marshaller marshaller = jaxbContext.createMarshaller();
            marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
            marshaller.marshal(vectors, System.out);

        } catch (JAXBException e) {
            e.printStackTrace();
        }
    }

}

文件.xml

<?xml version="1.0" encoding="UTF-8"?>
<Vectors>
    <Vector>
        <Id>1</Id>
        <Name>abc</Name>
         <Desc>abcdefg</Desc>
         <Type>body</Type>
         <Snippet>
            <![CDATA[<total>1234</total>]]>
         </Snippet>
    </Vector>
    <Vector>
         <Id>2</Id>
         <Name>def</Name>
         <Desc>abcdefghigj</Desc>
         <Type>body</Type>
         <Snippet>
            <![CDATA[<total>12345</total>]]>
         </Snippet>
     </Vector>
</Vectors>

输出

下面是运行演示代码的输出:

com.abc.TryXmltoJava.Vectors@44fe9319
<?xml version="1.0" encoding="UTF-8"?>
<Vectors>
   <Vector>
      <Id>1</Id>
      <Name>abc</Name>
      <Desc>abcdefg</Desc>
      <Type>body</Type>
   </Vector>
   <Vector>
      <Id>2</Id>
      <Name>def</Name>
      <Desc>abcdefghigj</Desc>
      <Type>body</Type>
   </Vector>
</Vectors>

【讨论】:

  • 非常感谢。顺便说一句,这意味着“@XmlAccessorType(XmlAccessType.FIELD)”
  • @Sanket - 默认情况下,JAXB (JSR-222) 实现将为所有公共字段创建元数据。如果您希望 JAXB (JSR-222) 实现将元数据基于字段,则可以指定 @XmlAccesorType(XmlAccessType.FIELD)。见:blog.bdoughan.com/2011/06/using-jaxbs-xmlaccessortype-to.html
【解决方案3】:

您是否可以先在 Java 中创建对象并使用 JAXB 持久保存对象,而不是尝试恢复对象?然后您将看到您的 XML 文件与 JAXB 生成的文件之间的区别。

【讨论】:

    【解决方案4】:

    首先,您的堆栈跟踪名称为 Vekctors(注意 k),因此可能是拼写错误。但随后它说“预期的元素是(无)”。在您的 Vectors 类中,您只声明了一个 Vector 元素,因此您不能在 Vectors 元素中放置两个元素。

    【讨论】:

      【解决方案5】:

      您可能希望查看与generate classes from an xsd 讨论房屋的这个答案。它还提供了一个资源链接以从原始 xml 生成 xsd。

      您的堆栈跟踪也提到了“Vkectors”——它们是源文件中的拼写错误吗?

      最后,您可能希望从 jaxb tutorial 开始,然后从那里开始。

      祝你好运!

      【讨论】:

        猜你喜欢
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 2011-02-15
        • 2012-04-21
        • 2010-09-20
        • 1970-01-01
        相关资源
        最近更新 更多