2011-12-06 25 views
4

私は、私の必要性に応じて動作させるために@XMLJavaTypeAdapterを持つことに苦労していました。私は私の理解に間違っている可能性があります。より良い方法で理解してください。JAXB @XmlJavaTypeAdapter usage

要件:たとえば : 私はマッピングのリストを輸送XMLを持っているが、以下のようにオブジェクト:JAXBを使用して

<transport> 
    <mappings> 
     <product>XXX</product> 
     <eventName>XXX</eventName> 
     <destination>XXX</destination> 
     <destinationType>XXX</destinationType> 
    </mappings> 
</transport> 

は、私がList<Mappings>オブジェクトとしてXMLを読み取ることができています。

メモリ内の表現がList<Mappings>Map<String,Mappings>であるようにしたいと思います。マップ内のキーはevent:productNameで、List<Mapping>ではなくコードマップで使用します。

私は少し@XmlJavaTypeAdapterの使用法と混同されており、理解できません。これをどのように達成することができますか?

この点に関するお手伝いをいただければ幸いです。

おかげ ラジ

答えて

7

注:私はEclipseLink JAXB (MOXy)リードとJAXB 2 (JSR-222)専門家グループのメンバーです。

あなたは、次の操作を実行するためにMOXYで@XmlPath拡張子を活用できます。

交通を

Transportクラスは@XmlPath拡張子を使用しています。 @XmlPathがなければ、mappingsというグルーピング要素がドキュメントに追加されます。

package forum8403623; 

import java.util.*; 
import javax.xml.bind.annotation.*; 
import javax.xml.bind.annotation.adapters.XmlJavaTypeAdapter; 
import org.eclipse.persistence.oxm.annotations.XmlPath; 

@XmlRootElement 
@XmlAccessorType(XmlAccessType.FIELD) 
public class Transport { 

    @XmlJavaTypeAdapter(MappingsAdapter.class) 
    @XmlPath(".") 
    private Map<String, Mapping> mappings = new HashMap<String, Mapping>(); 

} 

マッピング

package forum8403623; 

import javax.xml.bind.annotation.*; 

@XmlAccessorType(XmlAccessType.FIELD) 
@XmlType(propOrder={"product", "eventName", "destination", "destinationType"}) 
public class Mapping { 

    String product; 
    String eventName; 
    String destination; 
    String destinationType; 

} 

MappingsAdapter

次のクラスは、我々はXMLマッピングのために使用される表現にMap<String, Mapping>へ/から変換するための責任があります。

package forum8403623; 

import java.util.*; 
import java.util.Map.Entry; 
import javax.xml.bind.annotation.adapters.XmlAdapter; 

public class MappingsAdapter extends XmlAdapter<MappingsAdapter.AdaptedMap, Map<String, Mapping>>{ 

    @Override 
    public Map<String, Mapping> unmarshal(AdaptedMap v) throws Exception { 
     Map<String, Mapping> mappings = new HashMap<String, Mapping>(); 
     for(Mapping mapping : v.mappings) { 
      mappings.put(mapping.product, mapping); 
     } 
     return mappings; 
    } 

    @Override 
    public AdaptedMap marshal(Map<String, Mapping> v) throws Exception { 
     AdaptedMap adaptedMap = new AdaptedMap(); 
     for(Entry<String,Mapping> entry : v.entrySet()) { 
      adaptedMap.mappings.add(entry.getValue()); 
     } 
     return adaptedMap; 
    } 

    static class AdaptedMap { 
     public List<Mapping> mappings = new ArrayList<Mapping>(); 
    } 
} 

デモ

package forum8403623; 

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

public class Demo { 

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

     Unmarshaller unmarshaller = jc.createUnmarshaller(); 
     File xml = new File("src/forum8403623/input.xml"); 
     Transport transport = (Transport) unmarshaller.unmarshal(xml); 

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

} 

jaxb.properties

あなたはJAXBと呼ばれるファイルを追加する必要があり、あなたのJAXBプロバイダとしてMOXYを使用します。次のエントリを持つドメインモデルと同じパッケージのプロパティ:用

javax.xml.bind.context.factory = org.eclipse.persistence.jaxb.JAXBContextFactory 

詳細情報

+0

まず、おかげであなたの迅速な応答ウィット良い例です。私は本当にそれを感謝します。 上記の手順と同じ手順を繰り返し、いくつかのSystem.outステートメントを追加することで、Adapterのmarshall/unmarshallメソッドが呼び出されていることがわかります。残念なことに、出力は常に空のXMLです。 XMLを使用してUnmarhshalling TransportオブジェクトでMappingsAdapter.AdaptedMapリストがいっぱいになっていないようです。 私は間違っていることはありますか?お知らせ下さい。 もう一度、ご協力いただきありがとうございます。 Raj – Raj

+0

@Raj - このサンプルをそのまま使用するには、クラスパスにeclipselink.jarが必要です。あなたはhttp://www.eclipse.org/eclipselink/downloads/から入手することができます。 JAXBリファレンス実装を使用している場合は、次の記事が便利です:http://blog.bdoughan.com/2010/07/xmladapter-jaxbs-secret-weapon.html –

+0

+1の最初のブログリンク "XmlAdapter - JAXBの秘密の武器 " – KNU