java 如何使用 XStream 解组 Map

声明:本页面是StackOverFlow热门问题的中英对照翻译,遵循CC BY-SA 4.0协议,如果您需要使用它,必须同样遵循CC BY-SA许可,注明原文地址和作者信息,同时你必须将它归于原作者(不是我):StackOverFlow 原文地址: http://stackoverflow.com/questions/17086757/
Warning: these are provided under cc-by-sa 4.0 license. You are free to use/share it, But you must attribute it to the original authors (not me): StackOverFlow

提示:将鼠标放在中文语句上可以显示对应的英文。显示中英文
时间:2020-11-01 00:55:35  来源:igfitidea点击:

How to unmarshal Map using XStream

javamavenxstream

提问by carlspring

I cannot seem to be able to unmarshall a Map properly using XStream.

我似乎无法使用 XStream 正确解组 Map。

I have the following code:

我有以下代码:

Storage class

存储类

public class Storage
{

    @XStreamAsAttribute
    private String basedir;

    @XStreamAsAttribute
    private Map<String, Repository> repositories = new LinkedHashMap<String, Repository>();

...

}

Repository class

存储库类

public class Repository
{

    @XStreamAsAttribute
    private String name;

    @XStreamAsAttribute
    private String basedir;

    @XStreamAsAttribute
    private int policy;

    @XStreamAsAttribute
    private int layout;

    @XStreamAsAttribute
    private int type;

...

}

ConfigurationParser class

配置解析器类

public class ConfigurationParser
{

    public Storage parseConfiguration(String xmlFile)
            throws FileNotFoundException
    {
        FileInputStream fis = new FileInputStream(xmlFile);

        XStream xstream = new XStream();
        xstream.autodetectAnnotations(true);

        xstream.alias("storage", Storage.class);
        xstream.alias("repositories", Map.class);
        xstream.alias("repository", Repository.class);

        return (Storage) xstream.fromXML(fis);
    }

}

configuration.xml

配置文件

<storage>
    <basedir>storages/storage0</basedir>
    <repositories>
        <repository>
            <name>repository0</name>
            <policy>snapshots</policy>
            <layout>maven2</layout>
            <type>hosted</type>
        </repository>
    </repositories>
</storage>

Could somebody please give me some hints as to what is wrong. This is the error message:

有人可以给我一些关于什么是错误的提示。这是错误消息:

com.thoughtworks.xstream.converters.ConversionException: name : name
---- Debugging information ----
message             : name
cause-exception     : com.thoughtworks.xstream.mapper.CannotResolveClassException
cause-message       : name
class               : java.util.HashMap
required-type       : java.util.HashMap
converter-type      : com.thoughtworks.xstream.converters.collections.MapConverter
path                : /storage/repositories/repository/name
line number         : 5
class[1]            : org.foo.storage.Storage
converter-type[1]   : com.thoughtworks.xstream.converters.reflection.ReflectionConverter
version             : null
-------------------------------
    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)
    at com.thoughtworks.xstream.converters.reflection.AbstractReflectionConverter.unmarshallField(AbstractReflectionConverter.java:355)
    at com.thoughtworks.xstream.converters.reflection.AbstractReflectionConverter.doUnmarshal(AbstractReflectionConverter.java:306)
    at com.thoughtworks.xstream.converters.reflection.AbstractReflectionConverter.unmarshal(AbstractReflectionConverter.java:234)
    at com.thoughtworks.xstream.core.TreeUnmarshaller.convert(TreeUnmarshaller.java:72)
    at com.thoughtworks.xstream.core.AbstractReferenceUnmarshaller.convert(AbstractReferenceUnmarshaller.java:65)
    at com.thoughtworks.xstream.core.TreeUnmarshaller.convertAnother(TreeUnmarshaller.java:66)
    at com.thoughtworks.xstream.core.TreeUnmarshaller.convertAnother(TreeUnmarshaller.java:50)
    at com.thoughtworks.xstream.core.TreeUnmarshaller.start(TreeUnmarshaller.java:134)
    at com.thoughtworks.xstream.core.AbstractTreeMarshallingStrategy.unmarshal(AbstractTreeMarshallingStrategy.java:32)
    at com.thoughtworks.xstream.XStream.unmarshal(XStream.java:1058)
    at com.thoughtworks.xstream.XStream.unmarshal(XStream.java:1042)
    at com.thoughtworks.xstream.XStream.fromXML(XStream.java:922)
    at org.foo.configuration.ConfigurationParser.parseConfiguration(ConfigurationParser.java:36)
    at org.foo.configuration.ConfigurationParserTest.testParseConfiguration(ConfigurationParserTest.java:26)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at org.junit.runners.model.FrameworkMethod.runReflectiveCall(FrameworkMethod.java:45)
    at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:15)
    at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:42)
    at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:20)
    at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:263)
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:68)
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:47)
    at org.junit.runners.ParentRunner.run(ParentRunner.java:231)
    at org.junit.runners.ParentRunner.schedule(ParentRunner.java:60)
    at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:229)
    at org.junit.runners.ParentRunner.access
public Storage parseConfiguration(String xmlFile)
        throws FileNotFoundException
{
    FileInputStream fis = new FileInputStream(xmlFile);

    XStream xstream = new XStream();
    xstream.autodetectAnnotations(true);
    xstream.alias("storage", Storage.class);

    return (Storage) xstream.fromXML(fis);
}
0(ParentRunner.java:50) at org.junit.runners.ParentRunner.evaluate(ParentRunner.java:222) at org.junit.runners.ParentRunner.run(ParentRunner.java:300) at org.junit.runner.JUnitCore.run(JUnitCore.java:157) at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:74) at com.intellij.rt.execution.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:196) at com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.java:64) at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57) at com.intellij.rt.execution.application.AppMain.main(AppMain.java:120) Caused by: com.thoughtworks.xstream.mapper.CannotResolveClassException: name at com.thoughtworks.xstream.mapper.DefaultMapper.realClass(DefaultMapper.java:56) at com.thoughtworks.xstream.mapper.MapperWrapper.realClass(MapperWrapper.java:30) at com.thoughtworks.xstream.mapper.DynamicProxyMapper.realClass(DynamicProxyMapper.java:55) at com.thoughtworks.xstream.mapper.MapperWrapper.realClass(MapperWrapper.java:30) at com.thoughtworks.xstream.mapper.PackageAliasingMapper.realClass(PackageAliasingMapper.java:88) at com.thoughtworks.xstream.mapper.MapperWrapper.realClass(MapperWrapper.java:30) at com.thoughtworks.xstream.mapper.ClassAliasingMapper.realClass(ClassAliasingMapper.java:79) at com.thoughtworks.xstream.mapper.MapperWrapper.realClass(MapperWrapper.java:30) at com.thoughtworks.xstream.mapper.MapperWrapper.realClass(MapperWrapper.java:30) at com.thoughtworks.xstream.mapper.MapperWrapper.realClass(MapperWrapper.java:30) at com.thoughtworks.xstream.mapper.MapperWrapper.realClass(MapperWrapper.java:30) at com.thoughtworks.xstream.mapper.MapperWrapper.realClass(MapperWrapper.java:30) at com.thoughtworks.xstream.mapper.MapperWrapper.realClass(MapperWrapper.java:30) at com.thoughtworks.xstream.mapper.ArrayMapper.realClass(ArrayMapper.java:74) at com.thoughtworks.xstream.mapper.MapperWrapper.realClass(MapperWrapper.java:30) at com.thoughtworks.xstream.mapper.MapperWrapper.realClass(MapperWrapper.java:30) at com.thoughtworks.xstream.mapper.MapperWrapper.realClass(MapperWrapper.java:30) at com.thoughtworks.xstream.mapper.MapperWrapper.realClass(MapperWrapper.java:30) at com.thoughtworks.xstream.mapper.MapperWrapper.realClass(MapperWrapper.java:30) at com.thoughtworks.xstream.mapper.MapperWrapper.realClass(MapperWrapper.java:30) at com.thoughtworks.xstream.mapper.MapperWrapper.realClass(MapperWrapper.java:30) at com.thoughtworks.xstream.mapper.CachingMapper.realClass(CachingMapper.java:45) at com.thoughtworks.xstream.core.util.HierarchicalStreams.readClassType(HierarchicalStreams.java:29) at com.thoughtworks.xstream.converters.collections.AbstractCollectionConverter.readItem(AbstractCollectionConverter.java:70) at com.thoughtworks.xstream.converters.collections.MapConverter.putCurrentEntryIntoMap(MapConverter.java:86) at com.thoughtworks.xstream.converters.collections.MapConverter.populateMap(MapConverter.java:78) at com.thoughtworks.xstream.converters.collections.MapConverter.populateMap(MapConverter.java:72) at com.thoughtworks.xstream.converters.collections.MapConverter.unmarshal(MapConverter.java:67) at com.thoughtworks.xstream.core.TreeUnmarshaller.convert(TreeUnmarshaller.java:72) ... 42 more

Update 1:

更新 1:

Right, so... if I have:

是的,所以......如果我有:

ConfigurationParser class

配置解析器类

public class Repository
{

    @XStreamAlias(value = "name")
    private String name;

    @XStreamAlias (value = "basedir")
    private String basedir;

    @XStreamAlias (value = "policy")
    private String policy = RepositoryPolicyEnum.MIXED.getPolicy();

    @XStreamAlias (value = "layout")
    private String layout = RepositoryLayoutEnum.MAVEN_2.getLayout();

    @XStreamAlias (value = "type")
    private String type = RepositoryTypeEnum.HOSTED.getType();

...

}

Repository class

存储库类

<storage>
    <basedir>storages/storage0</basedir>
    <repository>
        <basedir>/here</basedir>
        <name>repository0</name>
        <policy>snapshots</policy>
        <layout>maven2</layout>
        <type>hosted</type>
    </repository>
    <repository>
        <basedir>/here2</basedir>
        <name>repository2</name>
        <policy>snapshots</policy>
        <layout>maven2</layout>
        <type>hosted</type>
    </repository>
</storage>

and:

和:

configuration.xml

配置文件

<storage>
    <basedir>storages/storage0</basedir>
    <repositories>
        <repository>
            <name>repository0</name>
            <policy>snapshots</policy>
            <layout>maven2</layout>
            <type>hosted</type>
        </repository>
    </repositories>
</storage>

It works. However, I would like the configure.xmlto look like:

有用。但是,我希望configure.xml看起来像:

XStream xstream = new XStream();
xstream.alias("storage", Storage.class);
xstream.alias("repository", Map.class);
xstream.registerConverter(new MapEntryConverter());
Storage storage = (Storage) xstream.fromXML(fis);
System.out.println(storage);

Any suggestions...?

有什么建议...?

回答by fGo

Try this one

试试这个

public static class MapEntryConverter implements Converter {

    public boolean canConvert(Class clazz) {
        return AbstractMap.class.isAssignableFrom(clazz);
    }

    public void marshal(Object value, HierarchicalStreamWriter writer, MarshallingContext context) {
        AbstractMap map = (AbstractMap) value;
        for (Object obj : map.entrySet()) {
            Entry entry = (Entry) obj;
            writer.startNode(entry.getKey().toString());
            writer.setValue(entry.getValue().toString());
            writer.endNode();
        }
    }

    public Object unmarshal(HierarchicalStreamReader reader, UnmarshallingContext context) {

      Repository repo = new Repository();
      while (reader.hasMoreChildren()) {
        reader.moveDown();
        String nodeName = reader.getNodeName();
        if ("name".equalsIgnoreCase(nodeName)) {
          repo.setName(reader.getValue());
        } else if ("policy".equalsIgnoreCase(nodeName)) {
          repo.setPolicy(reader.getValue());
        } else if ("layout".equalsIgnoreCase(nodeName)) {
          repo.setLayout(reader.getValue());
        } else if ("type".equalsIgnoreCase(nodeName)) {
          repo.setType(reader.getValue());
        }
        reader.moveUp();
  }
      return repo;
    }

}

and your converter

和你的转换器

public class Storage {
  private String basedir;
  private List<Repository> repositories = new ArrayList<Repository>();
  ...
}

public class Repository {
  private String name;
  private String policy;
  private String layout;
  private String type;
  ...
}

And my classes are simple pojos

我的课程很简单

##代码##