Java Language
JAXB
Szukaj…
Wprowadzenie
JAXB lub Java Architecture for XML Binding (JAXB) to platforma programowa, która pozwala programistom Java mapować klasy Java na reprezentacje XML. Strona ta zapozna czytelników z JAXB za pomocą szczegółowych przykładów dotyczących jego funkcji przewidzianych głównie do zestawiania i rozszyfrowywania obiektów Java w formacie xml i odwrotnie.
Składnia
JAXB.marshall (obiekt, fileObjOfXML);
Obiekt obj = JAXB.unmarshall (fileObjOfXML, nazwa_klasy);
Parametry
Parametr | Detale |
---|---|
fileObjOfXML | Obiekt File XML |
Nazwa klasy | Nazwa klasy z rozszerzeniem .class |
Uwagi
Za pomocą narzędzia XJC dostępnego w JDK można automatycznie wygenerować kod Java struktury xml opisanej w schemacie xml (plik .xsd
), patrz temat XJC .
Pisanie pliku XML (marshalling obiektu)
import javax.xml.bind.annotation.XmlRootElement;
@XmlRootElement
public class User {
private long userID;
private String name;
// getters and setters
}
Za pomocą adnotacji XMLRootElement
możemy oznaczyć klasę jako element główny pliku XML.
import java.io.File;
import javax.xml.bind.JAXB;
public class XMLCreator {
public static void main(String[] args) {
User user = new User();
user.setName("Jon Skeet");
user.setUserID(8884321);
try {
JAXB.marshal(user, new File("UserDetails.xml"));
} catch (Exception e) {
System.err.println("Exception occurred while writing in XML!");
} finally {
System.out.println("XML created");
}
}
}
marshal()
służy do zapisywania zawartości obiektu w pliku XML. Tutaj obiekt user
i nowy obiekt File
są przekazywane jako argumenty do funkcji marshal()
.
Po pomyślnym wykonaniu powoduje to utworzenie pliku XML o nazwie UserDetails.xml
w ścieżce klasy o poniższej treści.
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<user>
<name>Jon Skeet</name>
<userID>8884321</userID>
</user>
Odczytywanie pliku XML (unmarshalling)
Aby odczytać plik XML o nazwie UserDetails.xml
z poniższą zawartością
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<user>
<name>Jon Skeet</name>
<userID>8884321</userID>
</user>
Potrzebujemy klasy POJO o nazwie User.java
jak poniżej
import javax.xml.bind.annotation.XmlRootElement;
@XmlRootElement
public class User {
private long userID;
private String name;
// getters and setters
}
Tutaj stworzyliśmy zmienne i nazwę klasy zgodnie z węzłami XML. Aby zmapować je, używamy adnotacji XmlRootElement
na klasie.
public class XMLReader {
public static void main(String[] args) {
try {
User user = JAXB.unmarshal(new File("UserDetails.xml"), User.class);
System.out.println(user.getName()); // prints Jon Skeet
System.out.println(user.getUserID()); // prints 8884321
} catch (Exception e) {
System.err.println("Exception occurred while reading the XML!");
}
}
}
W tym przypadku do analizy pliku XML używana jest metoda unmarshal()
. Bierze nazwę pliku XML i typ klasy jako dwa argumenty. Następnie możemy użyć metod pobierających obiekt do wydrukowania danych.
Użycie XmlAdapter do wygenerowania pożądanego formatu xml
Gdy pożądany format XML różni się od modelu obiektowego Java, można użyć implementacji XmlAdapter do przekształcenia obiektu modelowego w obiekt w formacie xml i odwrotnie. W tym przykładzie pokazano, jak umieścić wartość pola w atrybucie elementu o nazwie pola.
public class XmlAdapterExample {
@XmlAccessorType(XmlAccessType.FIELD)
public static class NodeValueElement {
@XmlAttribute(name="attrValue")
String value;
public NodeValueElement() {
}
public NodeValueElement(String value) {
super();
this.value = value;
}
public String getValue() {
return value;
}
public void setValue(String value) {
this.value = value;
}
}
public static class ValueAsAttrXmlAdapter extends XmlAdapter<NodeValueElement, String> {
@Override
public NodeValueElement marshal(String v) throws Exception {
return new NodeValueElement(v);
}
@Override
public String unmarshal(NodeValueElement v) throws Exception {
if (v==null) return "";
return v.getValue();
}
}
@XmlRootElement(name="DataObject")
@XmlAccessorType(XmlAccessType.FIELD)
public static class DataObject {
String elementWithValue;
@XmlJavaTypeAdapter(value=ValueAsAttrXmlAdapter.class)
String elementWithAttribute;
}
public static void main(String[] args) {
DataObject data = new DataObject();
data.elementWithValue="value1";
data.elementWithAttribute ="value2";
ByteArrayOutputStream baos = new ByteArrayOutputStream();
JAXB.marshal(data, baos);
String xmlString = new String(baos.toByteArray(), StandardCharsets.UTF_8);
System.out.println(xmlString);
}
}
Automatyczna konfiguracja odwzorowania XML pola / właściwości (@XmlAccessorType)
Adnotacja @XmlAccessorType
określa, czy pola / właściwości będą automatycznie serializowane do @XmlAccessorType
XML. Uwaga: adnotacje pól i metod @XmlElement
, @XmlAttribute
lub @XmlTransient
mają pierwszeństwo przed ustawieniami domyślnymi.
public class XmlAccessTypeExample {
@XmlAccessorType(XmlAccessType.FIELD)
static class AccessorExampleField {
public String field="value1";
public String getGetter() {
return "getter";
}
public void setGetter(String value) {}
}
@XmlAccessorType(XmlAccessType.NONE)
static class AccessorExampleNone {
public String field="value1";
public String getGetter() {
return "getter";
}
public void setGetter(String value) {}
}
@XmlAccessorType(XmlAccessType.PROPERTY)
static class AccessorExampleProperty {
public String field="value1";
public String getGetter() {
return "getter";
}
public void setGetter(String value) {}
}
@XmlAccessorType(XmlAccessType.PUBLIC_MEMBER)
static class AccessorExamplePublic {
public String field="value1";
public String getGetter() {
return "getter";
}
public void setGetter(String value) {}
}
public static void main(String[] args) {
try {
System.out.println("\nField:");
JAXB.marshal(new AccessorExampleField(), System.out);
System.out.println("\nNone:");
JAXB.marshal(new AccessorExampleNone(), System.out);
System.out.println("\nProperty:");
JAXB.marshal(new AccessorExampleProperty(), System.out);
System.out.println("\nPublic:");
JAXB.marshal(new AccessorExamplePublic(), System.out);
} catch (Exception e) {
System.err.println("Exception occurred while writing in XML!");
}
}
} // outer class end
Wynik
Field:
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<accessorExampleField>
<field>value1</field>
</accessorExampleField>
None:
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<accessorExampleNone/>
Property:
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<accessorExampleProperty>
<getter>getter</getter>
</accessorExampleProperty>
Public:
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<accessorExamplePublic>
<field>value1</field>
<getter>getter</getter>
</accessorExamplePublic>
Ręczna konfiguracja odwzorowania XML pola / właściwości
Adnotacje @XmlElement
, @XmlAttribute
lub @XmlTransient
i inne w pakiecie javax.xml.bind.annotation
pozwalają programiście określić, w jaki sposób i w jaki sposób oznaczone pola lub właściwości powinny być serializowane.
@XmlAccessorType(XmlAccessType.NONE) // we want no automatic field/property marshalling
public class ManualXmlElementsExample {
@XmlElement
private String field="field value";
@XmlAttribute
private String attribute="attr value";
@XmlAttribute(name="differentAttribute")
private String oneAttribute="other attr value";
@XmlElement(name="different name")
private String oneName="different name value";
@XmlTransient
private String transientField = "will not get serialized ever";
@XmlElement
public String getModifiedTransientValue() {
return transientField.replace(" ever", ", unless in a getter");
}
public void setModifiedTransientValue(String val) {} // empty on purpose
public static void main(String[] args) {
try {
JAXB.marshal(new ManualXmlElementsExample(), System.out);
} catch (Exception e) {
System.err.println("Exception occurred while writing in XML!");
}
}
}
Określenie instancji XmlAdapter w celu (ponownego) wykorzystania istniejących danych
Czasami należy użyć określonych instancji danych. Odtwarzanie nie jest pożądane, a odniesienie do danych static
miałoby zapach kodu.
Możliwe jest określenie XmlAdapter
przykład Unmarshaller
należy używać, co pozwala użytkownikowi na korzystanie XmlAdapter
s bez zera-Arg konstruktora i / lub przekazywania danych do adaptera.
Przykład
Klasa użytkownika
Następująca klasa zawiera nazwę i obraz użytkownika.
import java.awt.image.BufferedImage;
import javax.xml.bind.annotation.XmlAttribute;
import javax.xml.bind.annotation.XmlRootElement;
import javax.xml.bind.annotation.adapters.XmlJavaTypeAdapter;
@XmlRootElement
public class User {
private String name;
private BufferedImage image;
@XmlAttribute
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@XmlJavaTypeAdapter(value=ImageCacheAdapter.class)
@XmlAttribute
public BufferedImage getImage() {
return image;
}
public void setImage(BufferedImage image) {
this.image = image;
}
public User(String name, BufferedImage image) {
this.name = name;
this.image = image;
}
public User() {
this("", null);
}
}
Adapter
Aby uniknąć dwukrotnego utworzenia tego samego obrazu w pamięci (a także ponownego pobrania danych), adapter przechowuje obrazy na mapie.
Aby uzyskać prawidłowy kod Java 7, zamień metodę getImage
public BufferedImage getImage(URL url) {
BufferedImage image = imageCache.get(url);
if (image == null) {
try {
image = ImageIO.read(url);
} catch (IOException ex) {
Logger.getLogger(ImageCacheAdapter.class.getName()).log(Level.SEVERE, null, ex);
return null;
}
imageCache.put(url, image);
reverseIndex.put(image, url);
}
return image;
}
import java.awt.image.BufferedImage;
import java.io.IOException;
import java.net.URL;
import java.util.HashMap;
import java.util.Map;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.imageio.ImageIO;
import javax.xml.bind.annotation.adapters.XmlAdapter;
public class ImageCacheAdapter extends XmlAdapter<String, BufferedImage> {
private final Map<URL, BufferedImage> imageCache = new HashMap<>();
private final Map<BufferedImage, URL> reverseIndex = new HashMap<>();
public BufferedImage getImage(URL url) {
// using a single lookup using Java 8 methods
return imageCache.computeIfAbsent(url, s -> {
try {
BufferedImage img = ImageIO.read(s);
reverseIndex.put(img, s);
return img;
} catch (IOException ex) {
Logger.getLogger(ImageCacheAdapter.class.getName()).log(Level.SEVERE, null, ex);
return null;
}
});
}
@Override
public BufferedImage unmarshal(String v) throws Exception {
return getImage(new URL(v));
}
@Override
public String marshal(BufferedImage v) throws Exception {
return reverseIndex.get(v).toExternalForm();
}
}
Przykładowe pliki XML
Poniższe 2 xmls są dla Jona Skeeta i jego odpowiednika dla Ziemi 2, które wyglądają dokładnie tak samo i dlatego używają tego samego awatara.
<?xml version="1.0" encoding="UTF-8"?>
<user name="Jon Skeet" image="https://www.gravatar.com/avatar/6d8ebb117e8d83d74ea95fbdd0f87e13?s=328&d=identicon&r=PG"/>
<?xml version="1.0" encoding="UTF-8"?>
<user name="Jon Skeet (Earth 2)" image="https://www.gravatar.com/avatar/6d8ebb117e8d83d74ea95fbdd0f87e13?s=328&d=identicon&r=PG"/>
Korzystanie z adaptera
ImageCacheAdapter adapter = new ImageCacheAdapter();
JAXBContext context = JAXBContext.newInstance(User.class);
Unmarshaller unmarshaller = context.createUnmarshaller();
// specifiy the adapter instance to use for every
// @XmlJavaTypeAdapter(value=ImageCacheAdapter.class)
unmarshaller.setAdapter(ImageCacheAdapter.class, adapter);
User result1 = (User) unmarshaller.unmarshal(Main.class.getResource("user.xml"));
// unmarshal second xml using the same adapter instance
Unmarshaller unmarshaller2 = context.createUnmarshaller();
unmarshaller2.setAdapter(ImageCacheAdapter.class, adapter);
User result2 = (User) unmarshaller2.unmarshal(Main.class.getResource("user2.xml"));
System.out.println(result1.getName());
System.out.println(result2.getName());
// yields true, since image is reused
System.out.println(result1.getImage() == result2.getImage());
Powiązanie przestrzeni nazw XML z serializowalną klasą Java.
To jest przykład pliku package-info.java
, który wiąże przestrzeń nazw XML z możliwą do serializacji klasą Java. Powinien być umieszczony w tym samym pakiecie co klasy Java, które powinny być serializowane przy użyciu przestrzeni nazw.
/**
* A package containing serializable classes.
*/
@XmlSchema
(
xmlns =
{
@XmlNs(prefix = MySerializableClass.NAMESPACE_PREFIX, namespaceURI = MySerializableClass.NAMESPACE)
},
namespace = MySerializableClass.NAMESPACE,
elementFormDefault = XmlNsForm.QUALIFIED
)
package com.test.jaxb;
import javax.xml.bind.annotation.XmlNs;
import javax.xml.bind.annotation.XmlNsForm;
import javax.xml.bind.annotation.XmlSchema;
Użycie XmlAdapter do przycięcia ciągu.
package com.example.xml.adapters;
import javax.xml.bind.annotation.adapters.XmlAdapter;
public class StringTrimAdapter extends XmlAdapter<String, String> {
@Override
public String unmarshal(String v) throws Exception {
if (v == null)
return null;
return v.trim();
}
@Override
public String marshal(String v) throws Exception {
if (v == null)
return null;
return v.trim();
}
}
I w package-info.java dodaj następującą deklarację.
@XmlJavaTypeAdapter(value = com.example.xml.adapters.StringTrimAdapter.class, type = String.class)
package com.example.xml.jaxb.bindings;// Packge where you intend to apply trimming filter
import javax.xml.bind.annotation.adapters.XmlJavaTypeAdapter;