我正在尝试使用XStream和丢弃来序列化和反序列化一个芭乐的多重映射。下面是一个简单的测试来说明:
final XStream xstream = new XStream(new JettisonMappedXmlDriver());
final Multimap<TestEnum, String> test = HashMultimap.create();
test.put(TestEnum.E1, "test");
final String json = xstream.toXML(test);
final Multimap<TestEnum, String> result = (Multimap<TestEnum, String>)xstream.fromXML(json);它会给出以下错误:
com.thoughtworks.xstream.converters.ConversionException: Could not call com.google.common.collect.HashMultimap.readObject() : com.test.Test$TestEnum cannot be cast to java.lang.Integer
---- Debugging information ----
message : Could not call com.google.common.collect.HashMultimap.readObject()
cause-exception : java.lang.ClassCastException
cause-message : com.test.Test$TestEnum cannot be cast to java.lang.Integer
class : com.google.common.collect.HashMultimap
required-type : com.google.common.collect.HashMultimap
converter-type : com.thoughtworks.xstream.converters.reflection.SerializableConverter
path : /com.google.common.collect.HashMultimap/com.google.common.collect.HashMultimap
line number : -1
version : 1.4.7
-------------------------------请注意,当与枚举键一起使用时,此错误尤其集中在Multimap上。如果我使用Map而不是multimap,就不会出现错误。如果我使用String而不是Enum作为键,就不会出现错误。而且,如果我序列化成XML而不是JSON (也就是说,在构造函数中没有"JettisonMappedXmlDriver“),它就能完美地工作。
有没有一个很好的解决方案?我目前正在使用一种变通方法,将我的multimap替换为集合地图,但我更愿意找到一种方法来保留multimap。
发布于 2021-03-31 12:50:20
使用XStream序列化多映射的解决方案是使用多映射转换器并注册到XStream,如下所示:
public class MultimapConverter extends AbstractCollectionConverter {
public MultimapConverter(Mapper mapper) {
super(mapper);
}
@Override
public boolean canConvert(@SuppressWarnings("rawtypes") Class clazz) {
return Multimap.class.isAssignableFrom(clazz);
}
@Override
@SuppressWarnings("unchecked")
public void marshal(
Object value, HierarchicalStreamWriter writer, MarshallingContext context) {
Multimap map = (Multimap) value;
for (Object key : map.keySet()) {
for (Object v : map.get(key)) {
if (v != null) {
writer.startNode("entry");
writer.addAttribute("key", key.toString());
writeCompleteItem(v, context, writer);
writer.endNode();
}
}
}
}
@Override
public Object unmarshal(HierarchicalStreamReader reader, UnmarshallingContext context) {
ArrayListMultimap<Object, Object> map = ArrayListMultimap.create();
while (reader.hasMoreChildren()) {
reader.moveDown();
String key = reader.getAttribute("key");
Object value = null;
while (reader.hasMoreChildren()) {
reader.moveDown();
value = readBareItem(reader, context, map);
reader.moveUp();
}
reader.moveUp();
if (value != null) {
map.put(key, value);
}
}
return map;
}
}然后将其注册到您拥有的XStream序列化程序中,例如:
XStream xstream = new XStream(new JettisonMappedXmlDriver());
xstream.registerConverter(new MultimapConverter(xstream.getMapper()));
xstream.allowTypeHierarchy(Multimap.class);
xstream.addDefaultImplementation(ArrayListMultimap.class, Multimap.class);至于反序列化上述转换器的错误,其工作方式与预期一致:
@Test
public void testSerializeWithEnum() {
XStream xstream = new XStream(new JettisonMappedXmlDriver());
xstream.registerConverter(new MultimapConverter(xstream.getMapper()));
xstream.allowTypeHierarchy(Multimap.class);
xstream.addDefaultImplementation(ArrayListMultimap.class, Multimap.class);
Multimap<TestEnum, String> test = HashMultimap.create();
test.put(TestEnum.E1, "test");
String json = xstream.toXML(test);
final Multimap<TestEnum, String> result = (Multimap<TestEnum, String>)xstream.fromXML(json);
}
public enum TestEnum {
E1
}如果在此处放置断点并调试result变量,您将看到JSON已使用枚举的值反序列化。
https://stackoverflow.com/questions/27502648
复制相似问题