使用xStream序列化BiMap

时间:2014-09-01 15:43:39

标签: java xml guava xstream bimap

我想用BiMap序列化xStream。由于我不喜欢xStream为BiMap自动生成的代码,我认为将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已经可以转换HashMaps了。奇怪的是,它只有在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>

谁能告诉我,我做错了什么?

1 个答案:

答案 0 :(得分:2)

您无需使用reader.getValue()方法拨打unmarshal

public Object unmarshal(final HierarchicalStreamReader reader, final UnmarshallingContext context) {
    final HashMap<?, ?> serializedMap = (HashMap<?, ?>) context.convertAnother(null, HashMap.class);
    return convertMapToBiMap(serializedMap);
}

这将使用空白地图。