4

我正在尝试将 Java 库中的 Java 对象转换为 XML 文件。但是,我遇到了这个问题:

A a = new A();

// initializing for a

JAXBContext jc = JAXBContext.newInstance("libraryA.A");

Marshaller marshaller = jc.createMarshaller();
marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
marshaller.marshal(a, System.out);

然后我得到了这个例外:

javax.xml.bind.JAXBException: "libraryA.a" doesnt contain ObjectFactory.class or jaxb.index
    at com.sun.xml.internal.bind.v2.ContextFactory.createContext(ContextFactory.java:186)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
    at java.lang.reflect.Method.invoke(Method.java:597)
    at javax.xml.bind.ContextFinder.newInstance(ContextFinder.java:128)
    at javax.xml.bind.ContextFinder.find(ContextFinder.java:290)
    at javax.xml.bind.JAXBContext.newInstance(JAXBContext.java:372)
    at javax.xml.bind.JAXBContext.newInstance(JAXBContext.java:337)
    at javax.xml.bind.JAXBContext.newInstance(JAXBContext.java:244)

如果我改变:JAXBContext jc = JAXBContext.newInstance("libraryA.a");

到:

JAXBContext jc = JAXBContext.newInstance(libraryA.A.class);

然后我有另一个例外:

com.sun.xml.internal.bind.v2.runtime.IllegalAnnotationsException: 2 counts of IllegalAnnotationExceptions

library.A is an interface, and JAXB can't handle interfaces.
    this problem is related to the following location:
        at library.A

library.A does not have a no-arg default constructor.
    this problem is related 

to the following location:
    at library.A
4

2 回答 2

14

背景信息(来自相关问题

根据您对我对上一个问题的回答所做的评论,域模型已经与 JAXB 一起使用。让您的客户端和服务器通过 XML 进行通信的最简单方法是在两端利用已经注释的模型。

我刚刚检查了我的客户的源代码。在此过程中,我们需要使用 javax.xml.bind.JAXBContext 和 javax.xml.bind.Marshaller 将一个从 java 对象生成的 xml 文件转换回 xml 文件。所以我的问题是可以将xml文件读回相同的java对象吗?然后我们可以使用 java 对象进行下一步。提前致谢!


更新

似乎您的问题是由于具有通过具有支持实现类的接口定义的域模型。下面我将演示如何使用 JAXB 实现(Metro、MOXy、JaxMe 等)来处理这个问题。

演示代码

import java.io.File;

import javax.xml.bind.JAXBContext;
import javax.xml.bind.Marshaller;
import javax.xml.bind.Unmarshaller;

public class Demo {

    public static void main(String[] args) throws Exception {
        JAXBContext jc = JAXBContext.newInstance(CustomerImpl.class);

        Unmarshaller unmarshaller = jc.createUnmarshaller();
        File xml = new File("input.xml");
        Customer customer = (Customer) unmarshaller.unmarshal(xml);

        Address address = customer.getAddress();
        System.out.println(address.getStreet());

        Marshaller marshaller = jc.createMarshaller();
        marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
        marshaller.marshal(customer, System.out);
    }

}

接口模型

以下接口代表我们的领域模型。这些接口在不用于引导 JAXBContext 时。

顾客

public interface Customer {

    public Address getAddress();

    public void setAddress(Address address);

}

地址

public interface Address {

    public String getStreet();

    public void setStreet(String street);

}

实现类

实现类将使用 JAXB 映射到 XML。

CustomerImpl

注意在CustomerImpl类中我们使用属性@XmlElement上的注解address来指定类型是AddressImpl

import javax.xml.bind.annotation.XmlElement;
import javax.xml.bind.annotation.XmlRootElement;

@XmlRootElement(name="customer")
public class CustomerImpl implements Customer {

    private Address address;

    @XmlElement(type=AddressImpl.class)
    public Address getAddress() {
        return address;
    }

    public void setAddress(Address address) {
        this.address = address;
    }

}

AddressImpl

public class AddressImpl implements Address {

    private String street;

    public String getStreet() {
        return street;
    }

    public void setStreet(String street) {
        this.street = street;
    }

}

输入.xml

<?xml version="1.0" encoding="UTF-8"?>
<customer>
    <address>
        <street>1 Any Street</street>
    </address>
</customer>
于 2011-05-11T14:10:11.107 回答
1

如果您不必使用 JAXB,也许您可​​以使用XStream

它对您可以序列化到/从 XML 的内容施加很少的限制,如果您不需要专门的 JAXB,它可能是合适的。

于 2011-05-11T13:44:07.120 回答