java - 同一 Java 对象上的不同 XML 映射/绑定(bind)

标签 java xml xml-binding oxm

我有一个 Java 应用程序可以与其他几个信息系统互操作

同一个对象可以根据所针对的信息系统映射到不同的XML文件中

我的问题是:是否有一个 Java 解决方案可以在同一对象上执行多个 XML 映射/绑定(bind)

类似于 Bean 验证组的东西,可以使用不同的验证配置文件验证对象

在 JAXB 风格中,它可能是这样的,例如:

// pseudocode
@XmlRootElement(name="person", , profile="profile1")
@XmlRootElement(name="individual", profile="profile2")
@XmlRootElement(name="human", profile="profile3")
public class Person {

    @XmlElement(name = "lastName", profile="profile1")
    @XmlElement(name = "surname", profile="profile2")
    @XmlElement(name = "famillyName", profile="profile3")
    private String lastName;

    //...
}

然后

    // pseudocode
    File file = new File("C:\\file.xml");
    JAXBContext jaxbContext = JAXBContext.newInstance(Person.class);
    Marshaller jaxbMarshaller = jaxbContext.createMarshaller("profile1");
    jaxbMarshaller.marshal(person, file);

    //...

    File file = new File("C:\\file.xml");
    JAXBContext jaxbContext = JAXBContext.newInstance(Person.class);
    Unmarshaller jaxbUnmarshaller = jaxbContext.createUnmarshaller("profile1");
    Person person = (Person) jaxbUnmarshaller.unmarshal(file);

也许可以用 JAXB 做这样的事情,但我不知道


编辑

@Martin Serrano 的回复为我提供了一些使用 JAXB 进行优化的线索。

一个抽象类来统治它们:

@XmlTransient
public abstract class APerson {

    protected String lastname;

    public APerson() {
    }

    public APerson(APerson p) {
        lastname = p.lastname;
    }

    public abstract String getLastname();

    public void setLastname(String lastname) {
        this.lastname = lastname;
    }

    @Override
    public String toString() {
        return this.getClass().getSimpleName() + "{" + lastname + "}";
    }
}

以及进行映射的具体类:

默认映射的 Person 类:

@XmlRootElement
public class Person extends APerson {

    public Person() {
    }

    public Person(APerson p) {
        super(p);
    }

    @Override
    public String getLastname() {
        return lastname;
    }
}

和其他用于替代映射的类:

@XmlRootElement
public class Human extends APerson {

    public Human() {
    }

    public Human(APerson p) {
        super(p);
    }

    @Override
    @XmlElement(name = "famillyName")
    public String getLastname() {
        return lastname;
    }
}


@XmlRootElement
public class Individual extends APerson{   

    public Individual() {
    }

    public Individual(APerson p) {
        super(p);
    }    

    @Override
    @XmlElement(name = "surname")
    public String getLastname() {
        return lastname;
    }
}

然后测试:

public class Main {

    public static void main(String[] args) throws Exception {
        Person person = new Person();
        person.setLastname("Doe");

        String fileName = "person.xml";
        marshal(person, fileName);
        person = unmarshal(Person.class, fileName);
        System.out.println(person);

        fileName = "human.xml";
        Human human = new Human(person);
        marshal(human, fileName);
        human = unmarshal(Human.class, fileName);
        System.out.println(human);
        person = new Person(human);
        System.out.println(person);

        fileName = "individual.xml";
        Individual individual = new Individual(person);
        marshal(individual, fileName);
        individual = unmarshal(Individual.class, fileName);
        System.out.println(individual);
        person = new Person(individual);
        System.out.println(person);
    }

    private static <P extends APerson> void marshal(P person, String fileName) throws JAXBException {
        File file = new File(fileName);
        JAXBContext context = JAXBContext.newInstance(person.getClass());
        Marshaller marshaller = context.createMarshaller();
        marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
        marshaller.marshal(person, file);
        marshaller.marshal(person, System.out);
    }

    private static <P extends APerson> P unmarshal(Class<P> cl, String fileName) throws JAXBException {
        File file = new File(fileName);
        JAXBContext context = JAXBContext.newInstance(cl);
        Unmarshaller unmarshaller = context.createUnmarshaller();
        P person = (P) unmarshaller.unmarshal(file);
        return person;
    }

}

结果:

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<person>
    <lastname>Doe</lastname>
</person>
Person{Doe}
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<human>
    <famillyName>Doe</famillyName>
</human>
Human{Doe}
Person{Doe}
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<individual>
    <surname>Doe</surname>
</individual>
Individual{Doe}
Person{Doe}

我发现最好的解决方案是具有不同的映射并避免代码重复,因此我们只需要使用映射来实现 getter。

让我们看看它是否在解决更大的问题!

但是虽然它比拥有 3 个不同的类要好得多,但它仍然不是同一类上的 3 个映射......

最佳答案

执行此类操作的最简单方法是为每个系统使用不同的外观。这将允许您更改 JAXB 反/序列化对象的方式。它还允许您根据所讨论的系统进行其他转换和调整。基本 Person 类将具有您的默认序列化,并且每个系统都可以在必要时使用适合它的外观。这基本上是关于 mapping interfaces 的非官方 jaxb 指南中概述的方法。 .

可以为每个系统注册这样的外观,允许您将系统特定的配置文件保留在每个核心类型的代码之外。

使用您的示例,核心界面将是:

@XmlRootElement(name="person")
public interface Person {
  @XmlElement(name = "lastName")
  private String lastName;
}

那么对于 System1,您将拥有:

@XmlRootElement(name="individual")
public class Individual implements Person {
}

另一个系统可能有:

@XmlRootElement(name="human")
public class Human implements Person {
}

关于java - 同一 Java 对象上的不同 XML 映射/绑定(bind),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29777186/

相关文章:

c# - 在没有标称类的情况下在 Silverlight 中绑定(bind) XML

java - jsp中获取请求url

java - 同步(this)和同步(objectReference)之间的区别

xml - 使用 XSLT 在 XML 文件中添加文本行

c# - 使用 < 和 > 解析 XML

xml - 导出到 XML 的基本标准?

java - JiBX:如何在我的代码中继续使用接口(interface)?

java - 无限嵌套的评论。如何维护堆栈历史记录?

java - 在 Java 中捕获错误时要做什么)

java - 将 JAXB 与包含许多相同元素的 XML 文件一起使用