使用snakeYaml解析根有地图的YAML文档

2022-01-14 00:00:00 yaml java snakeyaml

我想将 YAML 文档读取到自定义对象的映射(而不是映射,snakeYaml 默认会这样做).所以这个:

I want to read a YAML document to a map of custom objects (instead of maps, which snakeYaml does by default). So this:

19:
  typeID: 2
  limit: 300
20:
  typeID: 8
  limit: 100

将加载到如下所示的地图:

Would be loaded to a map which looks like this:

Map<Integer, Item>

项目在哪里:

class Item {
    private Integer typeId;
    private Integer limit;
}

我找不到使用snakeYaml 的方法,也找不到更好的库来完成这项任务.

I could not find a way to do this with snakeYaml, and I couldn't find a better library for the task either.

文档仅包含嵌套在其他对象中的地图/集合的示例,因此您可以执行以下操作:

The documentation only has examples with maps/collections nested inside other objects, so that you can do the following:

    TypeDescription typeDescription = new TypeDescription(ClassContainingAMap.class);
    typeDescription.putMapPropertyType("propertyNameOfNestedMap", Integer.class, Item.class);
    Constructor constructor = new Constructor(typeDescription);
    Yaml yaml = new Yaml(constructor);
    /* creating an input stream (is) */
    ClassContainingAMap obj = (ClassContainingAMap) yaml.load(is);

但是当 Map 格式位于文档的根目录时,我该如何定义它呢?

But how do I go about defining the Map format when it is at the root of the document?

推荐答案

你需要添加一个自定义构造函数.但是,在您的情况下,您不想注册项目"或项目列表"标签.

You need to add a custom Constructor. However, in your case you don't want register an "item" or "item-list" tag.

实际上,您希望将 Duck Typing 应用于您的 Yaml.这不是超级高效,但有一种相对简单的方法可以做到这一点.

In effect, you want to apply Duck Typing to your Yaml. It's not super efficient, but there is a relatively easy way to do this.

class YamlConstructor extends Constructor {
  @Override
  protected Object constructObject(Node node) {

    if (node.getTag() == Tag.MAP) {
        LinkedHashMap<String, Object> map = (LinkedHashMap<String, Object>) super
                .constructObject(node);
        // If the map has the typeId and limit attributes
        // return a new Item object using the values from the map
        ...
    }
     // In all other cases, use the default constructObject.
    return super.constructObject(node);

相关文章