通过 YAML 文件加载基于抽象类的对象

问题描述

我想从 yaml 文件中加载包含基于抽象类的对象数组列表的对象。我收到此错误消息:

线程“LWJGL 应用程序”中的异常无法为 JavaBean=com.myyaml.test.ImplementationOfExampleClass@7a358cc1 创建 property=arraylistofAbstractObjects

在“读者”中,第 1 行,第 1 列: dummyLong: 1 ^

java.lang.InstantiationException 在“读者”中,第 3 行,第 3 列: - dummyFloat:444 ^

YAML 文件

Text

Java 类:

dummyLong: 1
arraylistofAbstractObjects:
  - dummyFloat: 444
  - dummyDouble: 123
public abstract class ExampleClass {
    protected ArrayList<AbstractClass> arraylistofAbstractObjects;
    protected long dummyLong = 111;
    
    public ExampleClass() {
    }

    public void setArraylistofAbstractObjects(ArrayList<AbstractClass> arraylistofAbstractObjects) {
        this.arraylistofAbstractObjects = arraylistofAbstractObjects;
    }

    public void setDummyLong(long dummyLong) {
        this.dummyLong = dummyLong;
    }
}
public class ImplementationOfExampleClass extends ExampleClass {
    
    public ImplementationOfExampleClass() {
    }
}
public abstract class AbstractClass {
    private int dummyInt = 22;
    
    public AbstractClass() {
    }

    public void setDummyInt(int dummyInt) {
        this.dummyInt = dummyInt;
    }
}
public class FirstImplementationOfAbstractClass extends AbstractClass {
    float dummyFloat = 111f;
    
    public FirstImplementationOfAbstractClass() {
    }

    public void setDummyFloat(float dummyFloat) {
        this.dummyFloat = dummyFloat;
    }
}

我的猜测是 yaml 不知道要使用哪种抽象类实现。 FirstImplementationOfAbstractClass 或 SecondImplementationOfAbstractClass。是否可以使用此类类通过 yaml 加载对象?

解决方法

这只有在您告诉 YAML 处理器您想在 YAML 端实例化哪个类时才有可能。您可以使用标签执行此操作:

dummyLong: 1
arrayListOfAbstractObjects:
  - !first
    dummyFloat: 444
  - !second
    dummyDouble: 123

然后,您可以指示您的 YAML 处理器根据项目的标签正确处理项目。例如。使用 SnakeYAML,你就可以了

class MyConstructor extends Constructor {
    public MyConstructor() {
        this.yamlConstructors.put(new Tag("!first"),new ConstructFirst());
        this.yamlConstructors.put(new Tag("!second"),new ConstructSecond());
    }

    private class ConstructFirst extends AbstractConstruct {
        public Object construct(Node node) {
            // raw values,as if you would have loaded the content into a generic map.
            final Map<Object,Object> values = constructMapping(node);
            final FirstImplementationOfAbstractClass ret =
                    new FirstImplementationOfAbstractClass();
            ret.setDummyFloat(Float.parseFloat(values.get("dummyFloat").toString()));
            return ret;
        }
    }

    private class ConstructSecond extends AbstractConstruct {
        public Object construct(Node node) {
            final Map<Object,Object> values = constructMapping(node);
            final SecondImplementationOfAbstractClass ret =
                    new SecondImplementationOfAbstractClass();
            ret.setDummyFloat(Double.parseDouble(values.get("dummyFloat").toString()));
            return ret;
        }
    }
}

注意:加载内容时可以更智能,避免toString,直接处理节点内容;为了便于演示,我使用了一个愚蠢的实现。

然后,你使用这个构造函数:

Yaml yaml = new Yaml(new MyConstructor());
ExampleClass loaded = yaml.loadAs(input,ImplementationOfExampleClass.class);
,

Node 类是一种转换为 Java 数据对象的 YAML 文件。我在调试器下发现它包含字段 ArrayList<E> 值。其中包含带有 YAML 文件字段的 NodeTuple(例如 dummyFloat)。所以我必须在 constructMapping(node) 方法中转换我自己的每个字段,然后将它们设置为例如ConstructFirst.construct(Node node) 在构造的对象上。

编辑:

所以我必须在 constructMapping(node) 方法中转换我自己的每个字段,然后将它们设置在例如ConstructFirst.construct(Node node) 在构造的对象上。

需要将 param node 转换为 MappingNode。该方法继承自 BaseConstructor.constructMapping(MappingNode 节点)。 Flyx 没有添加那个演员,我不知道从哪里得到它。感谢帮助。现在它起作用了。但是我仍然在为嵌套的抽象类而苦苦挣扎。也许我需要帮助,但我会尽量处理好自己。

这个链接也可能有帮助: Polymorphic collections in SnakeYaml