将复杂的JSON反序列化为Java,类嵌套多层

13

我正试图将Cucumber的Json输出转换为单个Java对象。其中包含四层嵌套的对象,我在反序列化时遇到了问题。目前我正在使用Jackson,但也可以考虑其他建议。
以下是我的Json代码:

{
"line": 1,
"elements": [
  {
    "line": 3,
    "name": "Converteren centimeters naar voeten/inches",
    "description": "",
    "id": "applicatie-neemt-maten-in-cm-en-converteert-ze-naar-voet/inch,-en-vice-versa;converteren-centimeters-naar-voeten/inches",
    "type": "scenario",
    "keyword": "Scenario",
    "steps": [
      {
        "result": {
          "duration": 476796588,
          "status": "passed"
        },
        "line": 4,
        "name": "maak Maten-object aan met invoer in \"centimeters\"",
        "match": {
          "arguments": [
            {
              "val": "centimeters",
              "offset": 37
            }
          ],
          "location": "StepDefinition.maakMatenObjectAanMetInvoerIn(String)"
        },
        "keyword": "Given "
      },
      {
        "result": {
          "duration": 36319,
          "status": "passed"
        },
        "line": 5,
        "name": "ik converteer",
        "match": {
          "location": "StepDefinition.converteerMaten()"
        },
        "keyword": "When "
      },
      {
        "result": {
          "duration": 49138,
          "status": "passed"
        },
        "line": 6,
        "name": "uitvoer bevat maat in \"voeten/inches\"",
        "match": {
          "arguments": [
            {
              "val": "voeten/inches",
              "offset": 23
            }
          ],
          "location": "StepDefinition.uitvoerBevatMaatIn(String)"
        },
        "keyword": "Then "
      }
    ]
  },
  {
    "line": 8,
    "name": "Converteren voeten/inches naar centimeters",
    "description": "",
    "id": "applicatie-neemt-maten-in-cm-en-converteert-ze-naar-voet/inch,-en-vice-versa;converteren-voeten/inches-naar-centimeters",
    "type": "scenario",
    "keyword": "Scenario",
    "steps": [
      {
        "result": {
          "duration": 84175,
          "status": "passed"
        },
        "line": 9,
        "name": "maak Maten-object aan met invoer in \"voeten/inches\"",
        "match": {
          "arguments": [
            {
              "val": "voeten/inches",
              "offset": 37
            }
          ],
          "location": "StepDefinition.maakMatenObjectAanMetInvoerIn(String)"
        },
        "keyword": "Given "
      },
      {
        "result": {
          "duration": 23928,
          "status": "passed"
        },
        "line": 10,
        "name": "ik converteer",
        "match": {
          "location": "StepDefinition.converteerMaten()"
        },
        "keyword": "When "
      },
      {
        "result": {
          "duration": 55547,
          "status": "passed"
        },
        "line": 11,
        "name": "uitvoer bevat maat in \"centimeters\"",
        "match": {
          "arguments": [
            {
              "val": "centimeters",
              "offset": 23
            }
          ],
          "location": "StepDefinition.uitvoerBevatMaatIn(String)"
        },
        "keyword": "Then "
      }
    ]
  }
],
"name": "Applicatie neemt maten in cm en converteert ze naar voet/inch, en vice versa",
"description": "",
"id": "applicatie-neemt-maten-in-cm-en-converteert-ze-naar-voet/inch,-en-vice-versa",
"keyword": "Feature",
"uri": "sample.feature"
}

我尝试了多种不同的方法。首先,我使用了嵌套内部类,但似乎你必须使它们静态化,我担心这样做不会起作用,因为我在一个对象中有多个相同的实例(例如,根目录中有多个“元素”对象)。然后我尝试将它们放入具有Json注释的单独类中。这就是我的结果(省略了setter):

public class CucumberUitvoer {
    private String name;
    private String description;
    private String id;
    private String keyword;
    private String uri;
    private int line;
    @JsonProperty("elements")
    private List<FeatureObject> elements;

    public CucumberUitvoer(){}
}

public class FeatureObject {
    private String name;
    private String description;
    private String id;
    private String type;
    private String keyword;
    private int line;
    @JsonProperty("steps")
    private List<StepObject> steps;

    public FeatureObject() {
    }
}

public class StepObject {
    @JsonProperty("result")
    private ResultObject result;
    private String name;
    private String given;
    private String location;
    private String keyword;
    private int line;
    @JsonProperty("match")
    private MatchObject match;

    public StepObject(){}
}

public class ResultObject {
    private int duration;
    private String status;

    public ResultObject(){}
}

public class MatchObject {
    @JsonProperty("arguments")
    private List<ArgumentObject> arguments;
    private String location;

    public MatchObject(){}
}

public class ArgumentObject {
    private String val;
    private String offset;

    public ArgumentObject(){}
}

为了澄清,这里有一个类图展示嵌套的工作原理。

这个解决方案给我返回以下错误:

com.fasterxml.jackson.databind.JsonMappingException: Can not deserialize instance of nl.icaprojecten.TestIntegratieQuintor.JSONInterpreter.CucumberUitvoer out of START_ARRAY token

这里是实际进行映射的代码:

ObjectMapper mapper = new ObjectMapper();
    mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);

    CucumberUitvoer obj1 = null;
    try {
        obj1 = mapper.readValue(json, CucumberUitvoer.class);
    } catch (IOException e) {
        e.printStackTrace();
    }

有没有快速的解决方法让这种方法能够工作,还是我应该尝试完全不同的方法?


你尝试过添加getter和setter吗?这些字段不是公共的,因此对象无法序列化到这些类中。 - Turbut Alin
1
我以为只需要setter?我有那些,只是在这里省略了以使代码更易读。 - KeizerHarm
@TurbutAlin 添加getter方法没有任何效果,恐怕是这样。 - KeizerHarm
静态类并不意味着你不能拥有多个实例。 - shmosel
@shmosel 这已经是一年前的事了,但还是谢谢 :) - KeizerHarm
显示剩余5条评论
3个回答

5

好的,我花了一些时间进行调试并试图找出问题所在,最后发现是一个非常明显的问题。

implements Serializable

这是我添加到MatchObject并奏效的行。

当我们尝试反序列化某些对象时,首先必须使这些类实现Serializable接口。


这让我取得了一些实际的进展,谢谢!但是现在我又遇到了一个新的错误:"java.lang.NoSuchMethodError: com.fasterxml.jackson.databind.JavaType.isReferenceType()Z"。 - KeizerHarm
@KeizerHarm 可能你忘记了 getter/setter 方法,或者你尝试反序列化的类型是错误的。例如,将“hi”存储到整数属性中。 - cralfaro

1
我刚刚尝试了您的示例代码,很奇怪,它可以工作。
请再次检查您的导入,确保JSON按提供的方式传入,并且确实有getter、setter和构造函数。

构造函数应该用this.var = var吗?我认为它们应该是空的。 - KeizerHarm
只返回翻译后的文本:使用空构造函数和设置器或完整构造函数(设置所有字段)。 - Turbut Alin
如@TurbutAlin所说,所有类。 - cralfaro
1
@TurbutAlin 我试过了,但由于我的声望不到15,它们不会被显示出来。很遗憾。 - KeizerHarm
1
非常感谢你们两位的帮助!我只是还不想“关闭”这个问题,因为我还不确定新错误是否与旧错误有关。Jackson版本是2.6.3,正如你所指定的那样。 - KeizerHarm
显示剩余14条评论

-1

你可以从这段代码中获取反序列化的想法,

public class testCustomDeSerializer extends JsonDeserializer<test> {

public testCustomDeSerializer() {
    this(null);
}

public TestCustomDeSerializer(Class t) {
   // super(t);
}

@Override
public Test deserialize(JsonParser p, DeserializationContext ctx) throws IOException, JsonProcessingException {
    ObjectCodec objectCodec = p.getCodec();
    JsonNode node = objectCodec.readTree(p);

    ObjectMapper objectMapper =  new ObjectMapper();
    Test test= new Test();


    test.setId(node.get("line").asText());

    List<elements> elementList = new ArrayList<>();
    JsonNode elementsNode = node.get("elements");
    Iterator<JsonNode> slaidsIterator = elementsNode.elements();
    while (slaidsIterator.hasNext()) {
        Steps steps= new Steps();
        JsonNode slaidNode = slaidsIterator.next();
        JsonNode stepNode= (JsonNode) slaidNode.get("Steps");
        BoundingPoly in = objectMapper.readValue(stepNode.toString(), Steps.class);
        elementsNode.setSteps(in);
        /// continue

 return
        }

希望能有所帮助


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接