我想使用NamedMapConverter的某种变体(来自XStream库)。

  new NamedMapConverter(xstream.getMapper(), "mapping", "value", String.class, "color", HexColor.class, true, true, xstream.getConverterLookup());


但我想将其用作注释。我的猜测在下面。用于HexColor的SingleValueConverter已实现。

  @XStreamConverter(value = NamedMapConverter.class, strings = { "mapping", "value", "color" }, booleans = { true, true }, types = { String.class, HexColor.class })
  private Map<String, HexColor> colorMappings;


但是,令人惊讶的是,它不起作用。我究竟做错了什么?下面的样本结果。

  <colorMappings>
        <mapping>
              <value>Something</value>
              <color>007cc2</color>
         </mapping>
  </colorMappings>


我发现了这种奇怪行为的原因。这是由XStream库中的错误引起的。在AnnotationMapper类中,有一个cacheConverter方法。

private Converter cacheConverter(final XStreamConverter annotation,
    final Class targetType) {
    Converter result = null;
    final Object[] args;
    final List<Object> parameter = new ArrayList<Object>();
    if (targetType != null && annotation.useImplicitType()) {
        parameter.add(targetType);
    }
    final List<Object> arrays = new ArrayList<Object>();
    arrays.add(annotation.booleans());
    arrays.add(annotation.bytes());
    arrays.add(annotation.chars());
    arrays.add(annotation.doubles());
    arrays.add(annotation.floats());
    arrays.add(annotation.ints());
    arrays.add(annotation.longs());
    arrays.add(annotation.shorts());
    arrays.add(annotation.strings());
    arrays.add(annotation.types());
    for(Object array : arrays) {
        if (array != null) {
            int length = Array.getLength(array);
            for (int i = 0; i < length; i++ ) {
                Object object = Array.get(array, i);
                if (!parameter.contains(object)) {
                    parameter.add(object);
                }
            }
        }
    }


密钥片段在下面重复。

                if (!parameter.contains(object)) {
                    parameter.add(object);
                }


结果,如果在任何一个数组中都有两个相似的值(在这种情况下-布尔值true和true,但其他重复值也会出现问题),则仅将其中一个添加到参数中。因此,参数列表将是错误的(缺少一个布尔值),并且与正确的构造函数不匹配。

但是,目前我还不知道如何解决该问题。

最佳答案

一个简单(但很丑陋)的解决方案是创建新的Converter来扩展NamedMapConverter

public class BetterNamedMapConverter extends NamedMapConverter {

    public BetterNamedMapConverter(Mapper mapper, String entryName, String keyName, Class keyType, String valueName, Class valueType,
            boolean asAttributes, ConverterLookup lookup) {
        super(mapper, entryName, keyName, keyType, valueName, valueType, asAttributes, asAttributes, lookup);
    }

}


并将注释更改为

@XStreamConverter(value = BetterNamedMapConverter.class, strings = { "mapping", "value", "color" }, booleans = { true }, types = { String.class, HexColor.class }, useImplicitType = false)


但是,该错误仍然使我感到困惑(在某些情况下,解决该问题可能需要更糟糕的hack)。

10-06 13:09