以下は、これを行う方法の例です。
レジストリ
5 つの異なる要素値のそれぞれに対して で@XmlRegistry
マークされたメソッドを持つ、 で注釈が付けられたクラスが必要になります。@XmlElementDecl
package forum11537931;
import javax.xml.bind.JAXBElement;
import javax.xml.bind.annotation.*;
import javax.xml.namespace.QName;
@XmlRegistry
public class Registry {
private static final String ARTIST = "artist";
private static final String FOO = "foo";
@XmlElementDecl(name=ARTIST)
public JAXBElement<Relation> createArtist(Relation relation) {
return new JAXBElement<Relation>(new QName(ARTIST), Relation.class, relation);
}
@XmlElementDecl(name=FOO, substitutionHeadName=ARTIST)
public JAXBElement<Relation> createFoo(Relation relation) {
return new JAXBElement<Relation>(new QName(FOO), Relation.class, relation);
}
}
関係アダプタ
を使用して、JSON 表現により適切にマップされるものにオブジェクトXmlAdapter
を変換します。これを行うために注釈をRelations
活用します ( http://blog.bdoughan.com/2010/12/represent-string-values-as-element.htmlを参照)。@XmlElementRef
package forum11537931;
import java.util.*;
import javax.xml.bind.*;
import javax.xml.bind.annotation.*;
import javax.xml.bind.annotation.adapters.XmlAdapter;
import javax.xml.namespace.QName;
public class RelationsAdapter extends XmlAdapter<RelationsAdapter.AdaptedRelations, Relations> {
@Override
public Relations unmarshal(AdaptedRelations v) throws Exception {
// TODO Auto-generated method stub
return null;
}
@Override
public AdaptedRelations marshal(Relations relations) throws Exception {
AdaptedRelations adaptedRelations = new AdaptedRelations();
for(Relation relation : relations.relations) {
adaptedRelations.relations.add(new JAXBElement<Relation>(new QName(relations.targetType), Relation.class, relation));
}
return adaptedRelations;
}
@XmlSeeAlso({Registry.class})
public static class AdaptedRelations {
@XmlElementRef(type=JAXBElement.class, name="artist")
public List<JAXBElement<Relation>> relations = new ArrayList<JAXBElement<Relation>>();
}
}
oxm.xml
を XML 表現に適用したくないので、XmlAdapter
MOXy の外部マッピング ドキュメントを使用して指定します ( http://blog.bdoughan.com/2010/12/extending-jaxb-representing-annotations.htmlを参照)。
<?xml version="1.0"?>
<xml-bindings
xmlns="http://www.eclipse.org/eclipselink/xsds/persistence/oxm"
package-name="forum11537931">
<java-types>
<java-type name="Work">
<java-attributes>
<xml-element java-attribute="relations">
<xml-java-type-adapter value="forum11537931.RelationsAdapter"/>
</xml-element>
</java-attributes>
</java-type>
</java-types>
</xml-bindings>
入力.xml
この例では、XML ドキュメントを単純化しました。
<?xml version="1.0" encoding="UTF-8"?>
<work id="4ff89cf0-86af-11de-90ed-001fc6f176ff">
<relation-list target-type="artist">
<relation type="composer">
<direction>backward</direction>
</relation>
</relation-list>
</work>
デモ
package forum11537931;
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 {
// XML
JAXBContext jc = JAXBContext.newInstance(Work.class, Registry.class);
Unmarshaller unmarshaller = jc.createUnmarshaller();
File xml = new File("src/forum11537931/input.xml");
Work work = (Work) unmarshaller.unmarshal(xml);
// JSON
Map<String, Object> properties = new HashMap<String, Object>(2);
properties.put(JAXBContextProperties.OXM_METADATA_SOURCE, "forum11537931/oxm.xml");
properties.put(JAXBContextProperties.MEDIA_TYPE, "application/json");
JAXBContext jsonJC = JAXBContext.newInstance(new Class[] {Work.class, Registry.class}, properties);
Marshaller jsonMarshaller = jsonJC.createMarshaller();
jsonMarshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
jsonMarshaller.marshal(work, System.out);
}
}
出力
{
"work" : {
"relations" : [ {
"artist" : [ {
"type" : "composer",
"direction" : "backward"
} ]
} ]
}
}
ドメインモデル
仕事
package forum11537931;
import java.util.List;
import javax.xml.bind.annotation.*;
@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class Work {
@XmlElement(name="relation-list")
List<Relations> relations;
}
関係
package forum11537931;
import java.util.*;
import javax.xml.bind.annotation.*;
@XmlAccessorType(XmlAccessType.FIELD)
public class Relations {
@XmlAttribute(name="target-type")
String targetType;
@XmlElement(name="relation")
List<Relation> relations = new ArrayList<Relation>();
}
関係
package forum11537931;
import javax.xml.bind.annotation.*;
@XmlAccessorType(XmlAccessType.FIELD)
public class Relation {
@XmlAttribute
String type;
String direction;
}
jaxb.properties
MOXy を JAXB プロバイダーとして指定するにはjaxb.properties
、次のエントリを使用して、ドメイン モデルと同じパッケージで呼び出されるファイルを含める必要があります ( http://blog.bdoughan.com/2011/05/specifying-eclipselink-moxy-asを参照)。 -your.html )
javax.xml.bind.context.factory=org.eclipse.persistence.jaxb.JAXBContextFactory