我有3个带注释的类(我仔细删除了getter,setter等,以使代码更具可读性):
Result.java
@XmlRootElement(name = "resultat")
@XmlType(propOrder = { "state", "content" })
public class Result {
protected State state;
protected Content content;
}
Content.java:
@XmlRootElement(name = "content")
@XmlSeeAlso({ Job.class })
public class Content {
}
和Job.java扩展了内容:
@XmlType(name = "job", propOrder = { "status" })
@XmlRootElement(name = "job")
public class JobStatus extends Content {
protected String status;
}
我将Job实例设置为结果类的内容成员的值,但是我得到以下生成的xml文件:
<result>
<state>
<tag>value</tag>
</state>
<content xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:type="job">
<status>ok</status>
</content>
</result>
但它不是一个有效的xml文件,我无法针对以下架构验证它,因为“xmlns:xsi =”http://www.w3.org/2001/XMLSchema-instance“xsi:type = “工作”“部分因为工作不是来自http://www.w3.org/2001/XMLSchema-instance的类型。
<xs:schema attributeFormDefault="unqualified" elementFormDefault="qualified" xmlns:xs="http://www.w3.org/2001/XMLSchema">
<xs:element name="result">
<xs:complexType>
<xs:sequence>
<xs:element name="state">
<xs:complexType>
<xs:sequence>
<xs:element type="xs:string" name="tag"/>
</xs:sequence>
</xs:complexType>
</xs:element>
<xs:element name="content">
<xs:complexType>
<xs:sequence>
<xs:element type="xs:string" name="status"/>
</xs:sequence>
</xs:complexType>
</xs:element>
</xs:sequence>
</xs:complexType>
</xs:element>
</xs:schema>
如何更改JAXB配置以避免此问题?
谢谢!
答案 0 :(得分:3)
问题不在于JAXB配置,而是在Schema与Java Classes的设计中。
Java类Content
不符合您的架构,因为元素status
是在架构中定义的,而不是在Java类中定义的。
另外,您没有在架构中定义job
。
属性xsi:type
用于继承。由于JobStatus类继承自Content,因此在XML中显示为content
类型job
。
我认为最好的解决方案是从架构中的status
中删除元素content
,然后在架构中定义job
,如下所示:
<complexType name="job">
<complexContent>
<extension base="content">
<sequence>
<xs:element type="xs:string" name="status"/>
</sequence>
</extension>
</complexContent>
</complexType>
答案 1 :(得分:2)
W A is correct您的JAXB实现完全按照您的要求执行(请参阅:http://blog.bdoughan.com/2010/11/jaxb-and-inheritance-using-xsitype.html)。如果您不想更改XML架构,请参阅下面的更改映射的方法。
<强>结果强>
我们将使用@XmlElement
字段上的content
注释来表明真实类型为Job
(请参阅:http://blog.bdoughan.com/2011/05/jaxb-and-interface-fronted-models.html)。
import javax.xml.bind.annotation.*;
@XmlRootElement
@XmlType(propOrder = { "state", "content" })
@XmlAccessorType(XmlAccessType.FIELD)
public class Result {
protected State state;
@XmlElement(type=Job.class)
protected Content content;
}
内容强>
我们将使用@XmlTransient
类上的Content
注释将其从继承层次结构中删除(请参阅:http://blog.bdoughan.com/2011/06/ignoring-inheritance-with-xmltransient.html)。
import javax.xml.bind.annotation.XmlTransient;
@XmlTransient
public class Content {
}
<强>作业强>
Job
正常映射。
import javax.xml.bind.annotation.*;
@XmlAccessorType(XmlAccessType.FIELD)
public class Job extends Content {
protected String status;
}
<强>演示强>
import java.io.File;
import javax.xml.bind.*;
public class Demo {
public static void main(String[] args) throws Exception {
JAXBContext jc = JAXBContext.newInstance(Result.class);
Unmarshaller unmarshaller = jc.createUnmarshaller();
File xml = new File("src/forum17993543/input.xml");
Result result = (Result) unmarshaller.unmarshal(xml);
Marshaller marshaller = jc.createMarshaller();
marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
marshaller.marshal(result,System.out);
}
}
<强> input.xml中/输出强>
<?xml version="1.0" encoding="UTF-8"?>
<result>
<state>
<tag>value</tag>
</state>
<content>
<status>ok</status>
</content>
</result>