/ / JAXB і стек підкласів створює неправильний json - json, jaxb, moxy

JAXB і стек підкласів створює неправильні json - json, jaxb, moxy

Чи можна домогтися зняття / маршалювання структури класу, яка використовує вкладеність кількох abstract класи?
Враховуючи структуру класу, як це:

@XmlAccessorType(XmlAccessType.FIELD)
public abstract class Animal {}
public abstract class Mammal extends Animal {}
public class Tiger extends Mammal {}
public class Elephant extends Mammal {}

The @XmlRootElementред Zoo клас має список тварин:

@XmlElementWrapper(name = "animals")
@XmlElements({
@XmlElement(name = "elephant", type = Elephant.class),
@XmlElement(name = "tiger", type = Tiger.class)
})
private List<Animal> animals;

Я думаю, ви зрозуміли ідею ... XML для цього:

<?xml version="1.0" encoding="utf-8"?>
<zoo>
<animals>
<tiger>
<name>Richard</name>
<furry>true</furry>
</tiger>
<elephant>
<name>Otis</name>
<furry>false</furry>
</elephant>
<tiger>
<name>Kirk</name>
<furry>true</furry>
</tiger>
</animals>
</zoo>

Це виглядає чудово, круто.
Тепер JSON ...

 {
"animals" : {
"tiger" : [ {
"name" : "Richard",
"furry" : true
}, {
"name" : "Kirk",
"furry" : true
} ],
"elephant" : [ {
"name" : "Otis",
"furry" : false
} ]
}
}

Чому це підгрупує Mammal об'єкти класу в JSON?

Я використовую EclipseLink MOXy 2.6 для маршінгу.

Відповіді:

1 для відповіді № 1

ОРИГІНАЛЬНИЙ ВІДПОВІДЬ

MOXy групує ключі tiger і elephant щоб їх не повторювати.


UPDATE # 1

Отже, неможливо отримати JSON на зразок {"тварини": [{"@type": "тигр"}, {"@type": слон "}, ...]}?

Так, це можливо, вам просто потрібно відобразити це так:

Зоопарк

import java.util.List;
import javax.xml.bind.annotation.*;

@XmlAccessorType(XmlAccessType.FIELD)
public class Zoo {

private List<Animal> animals;

}

Тварина

import javax.xml.bind.annotation.*;

@XmlSeeAlso({Elephant.class, Tiger.class})
@XmlAccessorType(XmlAccessType.FIELD)
public abstract class Animal {

}

Демонстрація

import java.util.*;
import javax.xml.bind.*;
import javax.xml.transform.stream.StreamSource;
import org.eclipse.persistence.jaxb.JAXBContextProperties;

public class Demo {

public static void main(String[] args) throws Exception {
Map<String, Object> properties = new HashMap<String, Object>();
properties.put(JAXBContextProperties.MEDIA_TYPE, "application/json");
properties.put(JAXBContextProperties.JSON_INCLUDE_ROOT, false);
properties.put(JAXBContextProperties.JSON_ATTRIBUTE_PREFIX, "@");
JAXBContext jc = JAXBContext.newInstance(new Class[] {Zoo.class}, properties);

Unmarshaller unmarshaller = jc.createUnmarshaller();
StreamSource json = new StreamSource("src/forum19384491/input.json");
Zoo zoo = unmarshaller.unmarshal(json, Zoo.class).getValue();

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

}

input.json/Output

{
"animals" : [ {
"@type" : "tiger"
}, {
"@type" : "elephant"
}, {
"@type" : "tiger"
} ]
}

UPDATE # 2

Якщо ви хочете зберегти своє поточне представлення XML і просто змінити представлення JSON, ви можете скористатися розширенням зовнішнього відображення документа MOXy (див .: http://blog.bdoughan.com/2010/12/extending-jaxb-representing-annotations.html)

Документ зіставлення (oxm.xml)

Ми використаємо зовнішній документ зіставлення MOXy, щоб змінити відображення для animals поле на Zoo клас

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

Демонстрація

У демо-коді нижче ми створюємо 2 екземпляри JAXBContext на тій самій моделі домену. Для JSON використовується зовнішній документ зіставлення для налаштування відображення. input.xml - це XML-документ із вашого запитання.

import java.io.File;
import java.util.*;
import javax.xml.bind.*;

import org.eclipse.persistence.jaxb.JAXBContextProperties;

public class Demo {

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

Unmarshaller unmarshaller = xmlJC.createUnmarshaller();
File xml = new File("src/forum19384491/input.xml");
Zoo zoo = (Zoo) unmarshaller.unmarshal(xml);

Map<String, Object> properties = new HashMap<String, Object>(4);
properties.put(JAXBContextProperties.OXM_METADATA_SOURCE, "forum19384491/oxm.xml");
properties.put(JAXBContextProperties.MEDIA_TYPE, "application/json");
properties.put(JAXBContextProperties.JSON_INCLUDE_ROOT, false);
properties.put(JAXBContextProperties.JSON_ATTRIBUTE_PREFIX, "@");
JAXBContext jsonJC = JAXBContext.newInstance(new Class[] {Zoo.class}, properties);

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

}

Вихідні дані

Нижче наведено результати запуску демо-коду.

{
"animals" : [ {
"@type" : "tiger",
"name" : "Richard",
"furry" : true
}, {
"@type" : "elephant",
"name" : "Otis",
"furry" : false
}, {
"@type" : "tiger",
"name" : "Kirk",
"furry" : true
} ]
}