gpt4 book ai didi

java - 使用 JAXB 支持略有变化的模式

转载 作者:搜寻专家 更新时间:2023-10-30 21:11:47 25 4
gpt4 key购买 nike

情况

我需要支持基于彼此之间仅略有不同的模式生成 XML 文档。具体来说,我需要支持的模式基于行业标准,随着时间的推移会略有变化,供应商可能会制作他们自己的定制版本。

问题

我打算使用具有继承性的 JAXB 2(来自 Metro)作为解决方案。我希望包结构最终是这样的:

    com.company.xml.schema.v1
com.company.xml.schema.v2
com.company.xml.schema.v2.vendorxyz

v2 包中的类将简单地扩展 v1 包中的类并在必要时覆盖。不幸的是,该计划最终无法实现,因为子类无法覆盖父类中的注释 (see here)。例如,如果模式中的属性在版本之间被重命名,则 v2 元素类将不得不完全重新实现该元素,而不是从 v1 继承。

所以据我所知只有两个选择

选项 1

为每个模式类型创建一个“基本”包,用@XmlAccessorType(XmlAccessType.NONE) 注释该包中的元素类,并删除所有其他注释。然后,在每个版本化包中创建子类化“基础”包中相应类的类,并添加所有必需的注释。这个解决方案在继承领域确实给了我一点帮助,但是代码重复很大,维护起来会是一个挑战。

选项 2

不要使用 JAXB。我真的不喜欢这个解决方案,因为我也想使用 JAX-RS/JAX-WS。

问题

  • 我应该如何使用 JAXB 以支持具有微小变化的多个模式,而无需大量代码重复?
  • 我应该关注其他技术组合吗?

编辑

Blaise 的以下解决方案非常适用于我们的大多数模式,这些模式只是彼此之间的一个小翻译,数据通常相同。但是,在使用带有包名称的继承来进行版本控制更有意义的情况下,我们遇到了一个问题。例如:

com.company.xml.schema.v1.ElementA
com.company.xml.schema.v2.ElementA

(其中 v2.ElementA 扩展 v1.ElementA)

在这种情况下使用 MOXy 的 OXM 会遇到错误,可以找到解决方法 here (使用 Blaise 提供的解决方案,同样如此!)

最佳答案

注意:我是 EclipseLink JAXB (MOXy) JAXB 2 (JSR-222) 的领导和成员专家组。

您可以使用 EclipseLink JAXB 中的外部绑定(bind)文档来映射 XML 模式之间的变化。

供应商 1

您可以使用标准 JAXB 注释来映射供应商之一:

package forum9419732;

import javax.xml.bind.annotation.*;

@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class Customer {

@XmlAttribute
private int id;

private String lastName;
private String firstName;

public int getId() {
return id;
}

public void setId(int id) {
this.id = id;
}

public String getLastName() {
return lastName;
}

public void setLastName(String lastName) {
this.lastName = lastName;
}

public String getFirstName() {
return firstName;
}

public void setFirstName(String firstName) {
this.firstName = firstName;
}

}

供应商 2

我们将使用 MOXy 的外部元数据来自定义注释提供的元数据。在下面的文档 (oxm-v2.xml) 中,我们将把 firstNamelastName 属性映射到 XML 属性:

<?xml version="1.0"?>
<xml-bindings
xmlns="http://www.eclipse.org/eclipselink/xsds/persistence/oxm"
package-name="forum9419732">
<java-types>
<java-type name="Customer">
<java-attributes>
<xml-attribute java-attribute="firstName"/>
<xml-attribute java-attribute="lastName"/>
</java-attributes>
</java-type>
</java-types>
</xml-bindings>

供应商 3

我们将再次使用 MOXy 的外部绑定(bind)文件 (oxm-v3.xml) 来覆盖注释。这次我们将使 id 属性成为 XML 元素。

<?xml version="1.0"?>
<xml-bindings
xmlns="http://www.eclipse.org/eclipselink/xsds/persistence/oxm"
package-name="forum9419732">
<java-types>
<java-type name="Customer">
<java-attributes>
<xml-element java-attribute="id" name="identifier"/>
</java-attributes>
</java-type>
</java-types>
</xml-bindings>

演示

下面的示例代码演示了如何指定外部元数据。请注意我是如何引入第四个供应商来展示可以组合外部元数据文档的。

package forum9419732;

import java.util.*;
import javax.xml.bind.*;
import org.eclipse.persistence.jaxb.JAXBContextFactory;

public class Demo {

public static void main(String[] args) throws JAXBException {
Customer customer = new Customer();
customer.setId(123);
customer.setFirstName("Jane");
customer.setLastName("Doe");

// VENDOR 1
JAXBContext jcV1 = JAXBContext.newInstance(Customer.class);
marshal(jcV1, customer);

// VENDOR 2
Map<String, Object> propertiesV2 = new HashMap<String, Object>(1);
propertiesV2.put(JAXBContextFactory.ECLIPSELINK_OXM_XML_KEY, "forum9419732/oxm-v2.xml");
JAXBContext jcV2 = JAXBContext.newInstance(new Class[] {Customer.class}, propertiesV2);
marshal(jcV2, customer);

// VENDOR 3
Map<String, Object> propertiesV3 = new HashMap<String, Object>(1);
propertiesV3.put(JAXBContextFactory.ECLIPSELINK_OXM_XML_KEY, "forum9419732/oxm-v3.xml");
JAXBContext jcV3 = JAXBContext.newInstance(new Class[] {Customer.class}, propertiesV3);
marshal(jcV3, customer);

// VENDOR 4
Map<String, Object> propertiesV4 = new HashMap<String, Object>(1);
List<String> oxmV4 = new ArrayList<String>(2);
oxmV4.add("forum9419732/oxm-v2.xml");
oxmV4.add("forum9419732/oxm-v3.xml");
propertiesV4.put(JAXBContextFactory.ECLIPSELINK_OXM_XML_KEY, oxmV4);
JAXBContext jcV4 = JAXBContext.newInstance(new Class[] {Customer.class}, propertiesV4);
marshal(jcV4, customer);
}

private static void marshal(JAXBContext jc, Customer customer) throws JAXBException {
Marshaller marshaller = jc.createMarshaller();
marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
marshaller.marshal(customer, System.out);
System.out.println();
}

}

输出

以下是每个供应商的输出。请记住,Customer 的同一个实例用于制作这些 XML 文档中的每一个。

<?xml version="1.0" encoding="UTF-8"?>
<customer id="123">
<lastName>Doe</lastName>
<firstName>Jane</firstName>
</customer>

<?xml version="1.0" encoding="UTF-8"?>
<customer id="123" lastName="Doe" firstName="Jane"/>

<?xml version="1.0" encoding="UTF-8"?>
<customer>
<identifier>123</identifier>
<lastName>Doe</lastName>
<firstName>Jane</firstName>
</customer>

<?xml version="1.0" encoding="UTF-8"?>
<customer lastName="Doe" firstName="Jane">
<identifier>123</identifier>
</customer>

了解更多信息

关于java - 使用 JAXB 支持略有变化的模式,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9419732/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com