JAXBContext.newInstance变体

时间:2013-05-31 15:25:00

标签: java xml jaxb xsd

我正在类 JAXBContext 中尝试各种形式的 newInstance 方法(我正在使用Oracle JDK 1.7附带的默认Sun JAXB实现)。

我不清楚何时可以将具体类与 ObjectFactory 类传递给 newInstance 方法。我应该注意到,我纯粹使用JAXB来解析XML文件,即仅在XML-> Java方向上。

这是展示我观点的绝对最小代码:

xsd文件

<?xml version="1.0" encoding="UTF-8"?>
<schema elementFormDefault="qualified"
    xmlns          ="http://www.w3.org/2001/XMLSchema"
    xmlns:a        ="http://www.example.org/A"
    targetNamespace="http://www.example.org/A">
    <element name="root" type="a:RootType"></element>

    <complexType name="RootType">
       <sequence>
           <element name="value" type="string"></element>
       </sequence>
    </complexType>
</schema>

鉴于上述XSD,以下 JAXBInstance.newInstance 调用成功创建了一个可解析样本 a.xml 文件的上下文:

  • jc = JAXBContext.newInstance(“example.a”);
  • jc = JAXBContext.newInstance(example.a.ObjectFactory.class);
  • jc = JAXBContext.newInstance(example.a.RootType.class,example.a.ObjectFactory.class);

但是,在运行时单独传递 example.a.RootType.class 会失败 javax.xml.bind.UnmarshalException

jc = JAXBContext.newInstance(example.a.RootType.class); // this fails at runtime.

任何人都能解开一些光明吗?我正在试验这些 JAXBContext :: newInstance 变体的原因是我偶然发现了this problem,其中接受的答案包括“基于个别类构建JAXB上下文而不是对象工厂“。我正在使用的示例 a.xml JAXB Java代码在帖子的末尾跟随。

示例a.xml使用

<?xml version="1.0" encoding="UTF-8"?>
<a:root xmlns:a="http://www.example.org/A"
        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
        xsi:schemaLocation="http://www.example.org/A A.xsd">
    <a:value>foo</a:value>
</a:root>

JAXB解析代码

public static void main (String args[]) throws JAXBException, FileNotFoundException {
    JAXBContext jc = null;
    message("using package context (press any key:)");
    jc = JAXBContext.newInstance("example.a");
    work(jc); // SUCCEEDS

    message("using Object factory (press any key):");
    jc = JAXBContext.newInstance(example.a.ObjectFactory.class);
    work(jc); // SUCCEEDS

    message("using class enumeration (press any key):");
    try {
        jc = JAXBContext.newInstance(example.a.RootType.class);
        work(jc);  // FAILS
    } catch (javax.xml.bind.UnmarshalException e) {
        e.printStackTrace();
    }

    message("using class enumeration and Object factory too (press any key):");
    jc = JAXBContext.newInstance(example.a.RootType.class, example.a.ObjectFactory.class);
    work(jc); // SUCCEEDS

}

private static void work(JAXBContext jc) throws JAXBException, FileNotFoundException {
    Unmarshaller u = jc.createUnmarshaller();
    RootType root = ((JAXBElement<RootType>)u.unmarshal( new FileInputStream( "a.xml" ))).getValue();
    System.out.println( root.getValue() );
}

1 个答案:

答案 0 :(得分:16)

从XML架构生成的JAXB模型

从XML模式生成的模型中创建JAXBContext时,我总是建议在生成的类的包名上执行此操作。

JAXBContext jc = JAXBContext.newInstance("example.a");

使用带有newInstance参数的ClassLoader方法更好。当您从Java SE迁移到Java EE环境时,这将为您带来悲伤。

JAXBContext jc = JAXBContext.newInstance("example.a", example.a.ObjectFactory.class.getClassLoader());

当您在包名称上创建JAXBContext时,JAXB impl假定您从XML模式生成模型并引入ObjectFactory类,因为它始终生成使用{{1注释的类用这个名字。

从Java模型开始

这时我建议人们使用接受课程的@XmlRegistry方法。从JAXB类引导newInstance时,名为JAXBContext的类没有什么特别之处。任何使用ObjectFactory注释的类都可以扮演ObjectFactory的角色,因此不会自动查找。{1}}。这就是为什么当你明确引用@XmlRegistry时你的用例有效,而当你没有引用时失败。