BiMapをxStreamでシリアライズしたいと思います。私は BiMap の xStream による自動生成コードが気に入らないので、BiMap を HashMap に変換して HashMap だけをシリアル化し、逆シリアル化するときに HashMap を再度読み込んで元に戻すのは良い考えだと思いました。 BiMapに。そこで、次のコンバーター戦略を思いつきました。
public class XStreamBiMapConverterExample
{
public void run()
{
XStream xStream = new XStream();
xStream.setMode( XStream.XPATH_ABSOLUTE_REFERENCES );
xStream.registerConverter( new BiMapConverter(), XStream.PRIORITY_VERY_HIGH );
final String xml = xStream.toXML( new ObjectToSerialize() );
System.out.println( xml );
xStream.fromXML( xml );//Reading does not work, if the BiMap of ObjectToSerialize is empty
}
public static void main( final String[] args )
{
new XStreamBiMapConverterExample().run();
}
}
class ObjectToSerialize
{
// Map<String, Integer> serializeMap = new HashMap<>();
BiMap<String, Integer> serializeMap = HashBiMap.create();
public ObjectToSerialize()
{
//If there is no Values, my Converter fails. With Value there is no Problem.
// serializeMap.put( "Hallo", 7 );
}
}
class BiMapConverter implements Converter
{
@Override
public boolean canConvert( @SuppressWarnings( "rawtypes" ) final Class type )
{
return BiMap.class.isAssignableFrom( type );
}
@Override
public void marshal( final Object source, final HierarchicalStreamWriter writer,
final MarshallingContext context )
{
final BiMap<?, ?> biMap = (BiMap<?, ?>) source;
final HashMap<?, ?> convertBiMapToHashMap = convertMapToHashMap( biMap );
context.convertAnother( convertBiMapToHashMap );
}
private <K, V> HashMap<K, V> convertMapToHashMap( final Map<K, V> map )
{
final HashMap<K, V> hashMap = new HashMap<>();
for ( Entry<K, V> entry : map.entrySet() )
{
hashMap.put( entry.getKey(), entry.getValue() );
}
return hashMap;
}
@Override
public Object unmarshal( final HierarchicalStreamReader reader, final UnmarshallingContext context )
{
final HashMap<?, ?> serializedMap =
(HashMap<?, ?>) context.convertAnother( reader.getValue(), HashMap.class );
return convertMapToBiMap( serializedMap );
}
private <K, V> BiMap<K, V> convertMapToBiMap( final Map<K, V> map )
{
final BiMap<K, V> biMap = HashBiMap.create();
for ( Entry<K, V> entry : map.entrySet() )
{
biMap.put( entry.getKey(), entry.getValue() );
}
return biMap;
}
}
xStream はすでに HashMap を変換できるため、これは問題なく機能します。奇妙なことに、BiMap 内に値がある場合にのみ機能します。BiMap が空の場合、データの非整列化中に次の例外が発生します。
Exception in thread "main" com.thoughtworks.xstream.converters.ConversionException: only START_TAG can have attributes END_TAG seen ...ize>\n <serializeMap class="com.google.common.collect.HashBiMap"/>... @2:62 : only START_TAG can have attributes END_TAG seen ...ize>\n <serializeMap class="com.google.common.collect.HashBiMap"/>... @2:62
---- Debugging information ----
message : only START_TAG can have attributes END_TAG seen ...ize>\n <serializeMap class="com.google.common.collect.HashBiMap"/>... @2:62
cause-exception : java.lang.IndexOutOfBoundsException
cause-message : only START_TAG can have attributes END_TAG seen ...ize>\n <serializeMap class="com.google.common.collect.HashBiMap"/>... @2:62
class : com.google.common.collect.HashBiMap
required-type : com.google.common.collect.HashBiMap
converter-type : BiMapConverter
path : /ObjectToSerialize/serializeMap
line number : 2
class[1] : ObjectToSerialize
converter-type[1] : com.thoughtworks.xstream.converters.reflection.ReflectionConverter
version : 1.4.6
-------------------------------
at com.thoughtworks.xstream.core.TreeUnmarshaller.convert(TreeUnmarshaller.java:79)
at com.thoughtworks.xstream.core.AbstractReferenceUnmarshaller.convert(AbstractReferenceUnmarshaller.java:65)
at com.thoughtworks.xstream.core.TreeUnmarshaller.convertAnother(TreeUnmarshaller.java:66)
...
コンバーターを使用した後に生成される出力 (BiMap が空の場合) は次のとおりです。
<ObjectToSerialize>
<serializeMap class="com.google.common.collect.HashBiMap"/>
</ObjectToSerialize>
誰が私が間違っているのか教えてもらえますか?