首页 > 解决方案 > 通过 YAML 文件加载基于抽象类的对象

问题描述

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

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

在“阅读器”第 1 行第 1 列中:dummyLong: 1 ^

'reader',第 3 行,第 3 列中的 java.lang.InstantiationException:-dummyFloat:444 ^

YAML 文件

dummyLong: 1
arrayListOfAbstractObjects:
  - dummyFloat: 444
  - dummyDouble: 123

Java 类:

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;
    }
}
public class SecondImplementationOfAbstractClass extends AbstractClass {
    double dummyDouble = 333f;
    
    public SecondImplementationOfAbstractClass() {
    }

    public void setDummyDouble(double dummyDouble) {
        this.dummyDouble = dummyDouble;
    }
}

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

标签: javayamlsnakeyaml

解决方案


这只有在您告诉 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);

推荐阅读