在使用层次结构编组摘要 class 后删除 xsi:type
Remove xsi:type after marshalling abstract class with hierarchy
如果有人帮助我,那就太好了。所以我有一个结构:
<letters>
<list>
<name>Simon</name>
<type>2</type>
<passengerName>Johny</passengerName>
<passengerSurname>Revelator</passengerSurname>
</list>
<list>
<name>someName</name>
<type>4</type>
<fileURL>someUrl</fileURL>
<specialNotes>specialNotes</specialNotes>
</list>
</letters>
对于这个结构,我写了 Java Classes,他们编组和解组它:
Class一个
包裹 com.edhex.testing;
import javax.xml.bind.annotation.*;
@XmlAccessorType(XmlAccessType.PROPERTY)
@XmlSeeAlso({B.class, C.class})
@XmlType
abstract public class A {
int type;
String name;
@XmlElement
public int getType() {
return type;
}
public void setType(int type) {
this.type = type;
}
@XmlElement
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
Class 来信:
@XmlRootElement
@XmlAccessorType(XmlAccessType.PROPERTY)
public class Letters {
List<A> list;
@XmlElement
public List<A> getList() {
return list;
}
public void setList(List<A> list) {
this.list = list;
}
}
Class乙:
@XmlRootElement
@XmlType
@XmlAccessorType(XmlAccessType.PROPERTY)
public class B extends A {
String fileURL;
String specialNotes;
@XmlElement
public String getFileURL() {
return fileURL;
}
public void setFileURL(String fileURL) {
this.fileURL = fileURL;
}
@XmlElement
public String getSpecialNotes() {
return specialNotes;
}
public void setSpecialNotes(String specialNotes) {
this.specialNotes = specialNotes;
}
}
Class C:
@XmlRootElement
@XmlAccessorType(XmlAccessType.PROPERTY)
public class C extends A {
String passengerName;
String passengerSurname;
@XmlElement
public String getPassengerName() {
return passengerName;
}
public void setPassengerName(String passengerName) {
this.passengerName = passengerName;
}
@XmlElement
public String getPassengerSurname() {
return passengerSurname;
}
public void setPassengerSurname(String passengerSurname) {
this.passengerSurname = passengerSurname;
}
}
一切看起来都很好,但在编组之后我是这样的:
public static void main(String[] args) {
C c = new C();
c.setName("Simon");
c.setType(2);
c.setPassengerName("Johny");
c.setPassengerSurname("Revelator");
B b = new B();
b.setType(4);
b.setFileURL("someUrl");
b.setSpecialNotes("specialNotes");
b.setName("someName");
List<A> list = new ArrayList<A>(2);
list.add(c);
list.add(b);
Letters letter = new Letters();
letter.setList(list);
try {
File file = new File("C:\file.xml");
JAXBContext jaxbContext = JAXBContext.newInstance(Letters.class);
Marshaller jaxbMarshaller = jaxbContext.createMarshaller();
jaxbMarshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
jaxbMarshaller.marshal(letter, file);
jaxbMarshaller.marshal(letter, System.out);
} catch (JAXBException e) {
e.printStackTrace();
}
}
我得到这样的输出:
<letters>
<list xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:type="c">
<name>Lasha</name>
<type>2</type>
<passengerName>Johny</passengerName>
<passengerSurname>Revelator</passengerSurname>
</list>
<list xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:type="b">
<name>someName</name>
<type>4</type>
<fileURL>someUrl</fileURL>
<specialNotes>specialNotes</specialNotes>
</list>
</letters>
问题: 如何删除每个列表标签中的那些:xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:type="b"?
我知道 JAXB 会通知我它使用的是什么类型,因为最初映射使用的是另一种类型 (A.class),但我不需要该信息。有人有什么想法吗?
看起来您可以使用泛型 - 以保持类型的泛型:Remove xsi:type, xmlns:xs, and xmlns:xsi from JAXB Generics
但我的回答是 - 不要删除 xmlns:xsi 声明。我知道将它放在顶部元素中会更好看。但是 - 这样它仍然有效 XML 具有有效且完整的类型标识。 XML 是为了让机器读取。如果没有 xsi:type 属性,对象和 XML 之间的映射将不再是确定性的(而且您也无法可靠地验证 XML)。
玩得开心..
加布里埃尔
如果您在 list
属性 上使用 @XmlElementRef
,则该项目的元素将基于引用的 class 的 @XmlRootElement
和那么你将不会获得 xsi:type
属性。
@XmlRootElement
@XmlAccessorType(XmlAccessType.PROPERTY)
public class Letters {
List<A> list;
@XmlElementRef
public List<A> getList() {
return list;
}
public void setList(List<A> list) {
this.list = list;
}
}
如果有人帮助我,那就太好了。所以我有一个结构:
<letters>
<list>
<name>Simon</name>
<type>2</type>
<passengerName>Johny</passengerName>
<passengerSurname>Revelator</passengerSurname>
</list>
<list>
<name>someName</name>
<type>4</type>
<fileURL>someUrl</fileURL>
<specialNotes>specialNotes</specialNotes>
</list>
</letters>
对于这个结构,我写了 Java Classes,他们编组和解组它: Class一个 包裹 com.edhex.testing;
import javax.xml.bind.annotation.*;
@XmlAccessorType(XmlAccessType.PROPERTY)
@XmlSeeAlso({B.class, C.class})
@XmlType
abstract public class A {
int type;
String name;
@XmlElement
public int getType() {
return type;
}
public void setType(int type) {
this.type = type;
}
@XmlElement
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
Class 来信:
@XmlRootElement
@XmlAccessorType(XmlAccessType.PROPERTY)
public class Letters {
List<A> list;
@XmlElement
public List<A> getList() {
return list;
}
public void setList(List<A> list) {
this.list = list;
}
}
Class乙:
@XmlRootElement
@XmlType
@XmlAccessorType(XmlAccessType.PROPERTY)
public class B extends A {
String fileURL;
String specialNotes;
@XmlElement
public String getFileURL() {
return fileURL;
}
public void setFileURL(String fileURL) {
this.fileURL = fileURL;
}
@XmlElement
public String getSpecialNotes() {
return specialNotes;
}
public void setSpecialNotes(String specialNotes) {
this.specialNotes = specialNotes;
}
}
Class C:
@XmlRootElement
@XmlAccessorType(XmlAccessType.PROPERTY)
public class C extends A {
String passengerName;
String passengerSurname;
@XmlElement
public String getPassengerName() {
return passengerName;
}
public void setPassengerName(String passengerName) {
this.passengerName = passengerName;
}
@XmlElement
public String getPassengerSurname() {
return passengerSurname;
}
public void setPassengerSurname(String passengerSurname) {
this.passengerSurname = passengerSurname;
}
}
一切看起来都很好,但在编组之后我是这样的:
public static void main(String[] args) {
C c = new C();
c.setName("Simon");
c.setType(2);
c.setPassengerName("Johny");
c.setPassengerSurname("Revelator");
B b = new B();
b.setType(4);
b.setFileURL("someUrl");
b.setSpecialNotes("specialNotes");
b.setName("someName");
List<A> list = new ArrayList<A>(2);
list.add(c);
list.add(b);
Letters letter = new Letters();
letter.setList(list);
try {
File file = new File("C:\file.xml");
JAXBContext jaxbContext = JAXBContext.newInstance(Letters.class);
Marshaller jaxbMarshaller = jaxbContext.createMarshaller();
jaxbMarshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
jaxbMarshaller.marshal(letter, file);
jaxbMarshaller.marshal(letter, System.out);
} catch (JAXBException e) {
e.printStackTrace();
}
}
我得到这样的输出:
<letters>
<list xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:type="c">
<name>Lasha</name>
<type>2</type>
<passengerName>Johny</passengerName>
<passengerSurname>Revelator</passengerSurname>
</list>
<list xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:type="b">
<name>someName</name>
<type>4</type>
<fileURL>someUrl</fileURL>
<specialNotes>specialNotes</specialNotes>
</list>
</letters>
问题: 如何删除每个列表标签中的那些:xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:type="b"? 我知道 JAXB 会通知我它使用的是什么类型,因为最初映射使用的是另一种类型 (A.class),但我不需要该信息。有人有什么想法吗?
看起来您可以使用泛型 - 以保持类型的泛型:Remove xsi:type, xmlns:xs, and xmlns:xsi from JAXB Generics
但我的回答是 - 不要删除 xmlns:xsi 声明。我知道将它放在顶部元素中会更好看。但是 - 这样它仍然有效 XML 具有有效且完整的类型标识。 XML 是为了让机器读取。如果没有 xsi:type 属性,对象和 XML 之间的映射将不再是确定性的(而且您也无法可靠地验证 XML)。
玩得开心.. 加布里埃尔
如果您在 list
属性 上使用 @XmlElementRef
,则该项目的元素将基于引用的 class 的 @XmlRootElement
和那么你将不会获得 xsi:type
属性。
@XmlRootElement
@XmlAccessorType(XmlAccessType.PROPERTY)
public class Letters {
List<A> list;
@XmlElementRef
public List<A> getList() {
return list;
}
public void setList(List<A> list) {
this.list = list;
}
}