当前位置: 首页 > 知识库问答 >
问题:

使用StdDeserializer Jackson 2.5处理多态性

江嘉悦
2023-03-14

我有三个从超级类(SensorData)继承的类

@JsonDeserialize(using = SensorDataDeserializer.class)
public abstract class SensorData {

}

public class HumiditySensorData extends SensorData {
}

public class LuminositySensorData extends SensorData {
}

public class TemperatureSensorData extends SensorData {
}

我想根据参数将一个json输入转换为其中一个类。我正在尝试使用Jackson StdDeserializer并创建一个自定义的反序列化程序

@Component
public class SensorDataDeserializer extends StdDeserializer<SensorData> {

    private static final long serialVersionUID = 3625068688939160875L;

    @Autowired
    private SensorManager sensorManager;

    private static final String discriminator = "name";

    public SensorDataDeserializer() {
        super(SensorData.class);
        SpringBeanProvider.getInstance().autowireBean(this);
    }

    @Override
    public SensorData deserialize(JsonParser parser,
            DeserializationContext context) throws IOException,
            JsonProcessingException {
        ObjectMapper mapper = (ObjectMapper) parser.getCodec();
        ObjectNode root = (ObjectNode) mapper.readTree(parser);
        ObjectNode sensor = (ObjectNode) root.get("data");
        String type = root.get(discriminator).asText();
        Class<? extends SensorData> clazz = this.sensorManager
                .getCachedSensorsMap().get(type).sensorDataClass();
        if (clazz == null) {
            // TODO should throw exception
            return null;
        }
        return mapper.readValue(sensor.traverse(), clazz);
    }
}

我的问题是,当我确定映射具体类的正确类型时,映射器再次调用自定义stdDeserializer。所以我需要一种方法来打破循环当我有正确的类型。stacktrace是下一个

java.lang.NullPointerException
at com.hp.psiot.mapping.SensorDataDeserializer.deserialize(SensorDataDeserializer.java:38)
at com.hp.psiot.mapping.SensorDataDeserializer.deserialize(SensorDataDeserializer.java:1)
at com.fasterxml.jackson.databind.ObjectMapper._readValue(ObjectMapper.java:3532)
at com.fasterxml.jackson.databind.ObjectMapper.readValue(ObjectMapper.java:1868)
at com.hp.psiot.mapping.SensorDataDeserializer.deserialize(SensorDataDeserializer.java:47)
at com.hp.psiot.mapping.SensorDataDeserializer.deserialize(SensorDataDeserializer.java:1)
at com.fasterxml.jackson.databind.ObjectMapper._readMapAndClose(ObjectMapper.java:3560)
at com.fasterxml.jackson.databind.ObjectMapper.readValue(ObjectMapper.java:2660)
at org.springframework.http.converter.json.AbstractJackson2HttpMessageConverter.readJavaType(AbstractJackson2HttpMessageConverter.java:205)
at org.springframework.http.converter.json.AbstractJackson2HttpMessageConverter.read(AbstractJackson2HttpMessageConverter.java:200)
at org.springframework.web.servlet.mvc.method.annotation.AbstractMessageConverterMethodArgumentResolver.readWithMessageConverters    (AbstractMessageConverterMethodArgumentResolver.java:138)
at org.springframework.web.servlet.mvc.method.annotation.RequestResponseBodyMethodProcessor.readWithMessageConverters(RequestResponseBodyMethodProcessor.java:184)
at org.springframework.web.servlet.mvc.method.annotation.RequestResponseBodyMethodProcessor.resolveArgument(RequestResponseBodyMethodProcessor.java:105)

输入示例

{
    "name":"temperature",
    "data": {
        "value":20
    }
}

我只包含stacktrace来显示映射器正在再次调用反序列化器。nullPointerException的原因是,当调用第二个ObjectMapper时,输入为

"value":20

因此,抛出了一个异常,因为我们没有确定类型的信息,也没有检查输入是否正确

如果可能的话,我希望避免使用JsonSubTypes和JsonTypeInfo。

提前道谢!

部分解

class ServiceData {
    @JsonDeserialize(using = SensorDataDeserializer.class)
    List<SensorData> sensors;

}
 @Component
 public class SensorDataDeserializer extends StdDeserializer<List<SensorData>> {

      private static final long serialVersionUID = 3625068688939160875L;

      @Autowired
      private SensorManager sensorManager;

      private static final String discriminator = "name";

      public SensorDataDeserializer() {
           super(SensorData.class);
            SpringBeanProvider.getInstance().autowireBean(this);
      }

      @Override
      public List<SensorData> deserialize(JsonParser parser,
                DeserializationContext context) throws IOException,
                JsonProcessingException {
           try {
                ObjectMapper mapper = (ObjectMapper) parser.getCodec();
                ArrayNode root = (ArrayNode) mapper.readTree(parser);
                int size = root.size();
                List<SensorData> sensors = new ArrayList<SensorData>();
                for (int i = 0; i < size; ++i) {
                     ObjectNode sensorHead = (ObjectNode) root.get(i);
                     ObjectNode sensorData = (ObjectNode) sensorHead.get("data");
                     String tag = sensorHead.get(discriminator).asText();
                     Class<? extends SensorData> clazz = this.sensorManager
                               .getCachedSensorsMap().get(tag).sensorDataClass();
                     if (clazz == null) {
                          throw new InvalidJson("unbound sensor");
                     }
                     SensorData parsed = mapper.readValue(sensorData.traverse(),
                               clazz);
                     if (parsed == null) {
                          throw new InvalidJson("unbound sensor");
                     }
                     sensors.add(parsed);
                }
                return sensors;
           } catch (Throwable e) {
                throw new InvalidJson("invalid data");
           }

      }
 }

希望它能帮助某人:)

共有1个答案

蔺翰音
2023-03-14

为什么不直接使用@jsonTypeInfo?多态处理是它的特定用例。

在这种情况下,您需要使用如下内容:

@JsonTypeInfo(use=Id.NAME, include=As.PROPERTY, property="name")
@JsonSubTypes({ HumiditySensorData.class, ... }) // or register via mapper
public abstract class SensorData { ... }

@JsonTypeName("temperature")
public class TemperaratureSensorData extends SensorData {
   public TemperaratureSensorData(@JsonProperty("data") JsonNode data) {
     // extract pieces out
  }
}

这将处理从“name”到子类型的解析,将“data”的内容绑定为jsonnode(或者,如果您愿意,可以使用mapobject或任何匹配的类型)。

 类似资料:
  • 问题内容: 我有一个与此类似的类层次结构: 像这样的json输入(我无法改变自己) 我正在像这样用Jackson解析Java中的JSON 我想使用JAVA / Jackson从JSON反序列化类A,B和其他类。JSON中没有嵌入类型信息(并且不能)。我不能在类上使用批注(我不拥有它们),我(相信)我不能使用混合包,因为可能有任意数量的类,例如A和B(混合包不是动态的)。好的是,反序列化代码知道哪个

  • 我有一个作业,我必须通过PApplet处理器显示“健身房机器”和它们的数据。我需要使用多态性和继承(跑步机是体操机,等等)。每个健身房机器必须有自己的绘制方法。我只是想不出如何避免NullPointerException。 这是我的“processingexample.java”文件中的代码,它将通过处理器实际绘制图像: 在过去的5个小时里,我一直无法弄清楚为什么会抱怨NullPointerExc

  • 我使用的是Spring Batch 2.1.8。释放我有一个文件,它由一些头信息和一些需要处理的记录组成。 我有一个使用面向块处理的步骤。该步骤包含ItemReader和ItemWriter的实现。ItemReader实现是线程安全的,而ItemWriter不是。 我想在处理(或写入)任何记录之前使用标题信息。在继续使用面向块的处理时,如何确保这一点? 建议的解决方案:一种解决方案可以是编写一个预

  • 我正试图将这些值以如下方式放入数组: 我不太清楚如何使用regex处理多行文本。regex是正确的方法吗?

  • 问题内容: 我有一个像这样的课程: 我想序列化它们的列表: 当我进行序列化时,动物属性的类型信息将丢失。有没有一种方法可以某种方式安装解析器侦听器,以便在遇到列表中的每个元素时提供正确的类进行反序列化?这就是手动提供描述类类型的字符串的想法。 谢谢 问题答案: Gson项目代码库中的RuntimeTypeAdapter据说可以很好地用于多态序列化和反序列化。我认为我尚未尝试使用它。有关更多信息,请

  • 问题内容: 我写了一个程序,可以总结如下: 实际代码(尤其是)要复杂得多。仅使用将其当作参数的这些值(意味着它不引用) 基本上,它将巨大的数据集加载到内存中并进行处理。输出的写操作委托给一个子进程(它实际上写到多个文件中,这需要很多时间)。因此,每次处理一个数据项时,它都会通过res_queue发送到子流程,然后该子流程根据需要将结果写入文件中。 子流程不需要访问,读取或修改以任何方式加载的数据。