猿问

使用 objectMapper 将 JSON 日期格式反序列化为 ZonedDateTime

背景


我有以下 JSON(来自 Kafka 的消息)

{

      "markdownPercentage": 20,

      "currency": "SEK",

      "startDate": "2019-07-25"

}

我有以下(生成的 JSON 模式)POJO(我无法更改 POJO,因为它是公司的共享资源)

public class Markdown {

    @JsonProperty("markdownPercentage")

    @NotNull

    private Integer markdownPercentage = 0;

    @JsonProperty("currency")

    @NotNull

    private String currency = "";

    @JsonFormat(

        shape = Shape.STRING,

        pattern = "yyyy-MM-dd"

    )

    @JsonProperty("startDate")

    @NotNull

    private ZonedDateTime startDate;


    // Constructors, Getters, Setters etc.


}

我们的应用程序是一个 Spring Boot 应用程序,它使用 Spring Cloud Stream 从 Kafka 读取 JSON 消息 (1) 并使用 POJO (2) 然后对其进行处理。

问题


当应用程序尝试将消息反序列化为对象时,它会抛出异常

当前代码


我定义了以下 objectMapper


/**

     * Date mapper.

     *

     * @return the {@link ObjectMapper}

     */

    @Bean

    public ObjectMapper objectMapper() {

        ObjectMapper mapper = new ObjectMapper();

        mapper.registerModule(new JavaTimeModule());

        mapper.setDateFormat(new SimpleDateFormat("yyyy-MM-dd"));

        return mapper;

    }

问题


我知道 POJO 中生成的 ZonedDateTime 需要源消息中不存在的“时间”元素。我只能控制 objectMapper。是否有任何可能的配置可以使这项工作?


笔记


如果反序列化 POJO 中的时间元素“假定”为 startOfDay,即“00.00.00.000Z”,我很好


红颜莎娜
浏览 295回答 5
5回答

交互式爱情

我只能控制ObjectMapper. 是否有任何可能的配置可以使这项工作?只要您对时间和时区的默认值感到满意,就可以使用自定义反序列化器解决它:public class ZonedDateTimeDeserializer extends JsonDeserializer<ZonedDateTime> {&nbsp; &nbsp; @Override&nbsp; &nbsp; public ZonedDateTime deserialize(JsonParser jsonParser,&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;DeserializationContext deserializationContext)&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;throws IOException {&nbsp; &nbsp; &nbsp; &nbsp; LocalDate localDate = LocalDate.parse(&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; jsonParser.getText(),&nbsp;&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; DateTimeFormatter.ISO_LOCAL_DATE);&nbsp; &nbsp; &nbsp; &nbsp; return localDate.atStartOfDay(ZoneOffset.UTC);&nbsp; &nbsp; }}然后将其添加到模块并将模块注册到您的ObjectMapper实例:SimpleModule module = new SimpleModule();module.addDeserializer(ZonedDateTime.class, new ZonedDateTimeDeserializer());ObjectMapper mapper = new ObjectMapper();mapper.registerModule(module);如果将反序列化器添加到模块中不适合您(从某种意义上说,此配置将应用于其他ZonedDateTime实例),那么您可以依靠混合来定义反序列化器将应用于哪些字段。首先定义一个mix-in接口,如下图:public interface MarkdownMixIn {&nbsp; &nbsp; @JsonDeserialize(using = ZonedDateTimeDeserializer.class)&nbsp; &nbsp; ZonedDateTime getDate();}然后将混合接口绑定到所需的类:ObjectMapper mapper = new ObjectMapper();mapper.addMixIn(Markdown.class, MarkdownMixIn.class);

长风秋雁

问题:我想将日期从 json 解析为 java LocalDateTime/ZonedDateTime 对象。ZonedDateTimeSerializer 存在但 ZonedDateTimeDeserializer 不存在。因此,为什么我创建了一个自定义 ZonedDateTimeDeserializer。  public static final String ZONED_DATE_TIME_FORMAT = "yyyy-MM-dd'T'HH:mm:ss.SSSz";   @Getter  @Setter  @JsonSerialize(using = ZonedDateTimeSerializer.class)  @JsonDeserialize(using = ZonedDateTimeDeserializer.class) // Doesn't exist, So I created a custom ZonedDateDeserializer utility class.  @JsonFormat(pattern = ZONED_DATE_TIME_FORMAT)  @JsonProperty("lastUpdated")  private ZonedDateTime lastUpdated;解决方案:我最终得到了更简单、更少的代码行。用于反序列化ZonedDateTime的实用程序类:/** * Custom {@link ZonedDateTime} deserializer. * * @param jsonParser             for extracting the date in {@link String} format. * @param deserializationContext for the process of deserialization a single root-level value. * @return {@link ZonedDateTime} object of the date. * @throws IOException throws I/O exceptions. */public class ZonedDateTimeDeserializer extends JsonDeserializer<ZonedDateTime> {    @Override    public ZonedDateTime deserialize(JsonParser jsonParser, DeserializationContext deserializationContext)            throws IOException {        return ZonedDateTime.parse(jsonParser.getText(), DateTimeFormatter.ofPattern(ZONED_DATE_TIME_FORMAT));    }}如果您改用LocalDateTime怎么办。在那种情况下,它甚至更容易,反序列化器和序列化器类都已经提供给我们了。不需要上面定义的自定义实用程序类:  @Getter  @Setter  @JsonSerialize(using = LocalDateSerializer.class)  @JsonDeserialize(using = LocalDateTimeDeserializer.class)  @JsonFormat(pattern = ZONED_DATE_TIME_FORMAT) //Specify the format you want: "yyyy-MM-dd'T'HH:mm:ss.SSS"  @JsonProperty("created")  private LocalDateTime created;关键词:json格式 localDateTime zonedDateTime

Qyouu

不幸的是,默认情况下您不能反序列化为String Object格式ZonedDateTime。但是您可以通过两种方式克服这个问题。方式01ZonedDateTime将类型更改为LocalDate在您的类中键入POJO并将值作为字符串传递03-06-2012方式02但是如果您必须使用时区存储日期和时间,那么您必须执行以下方法来克服步骤01ZonedDateTime使用 DateTimeFormat创建反序列化类public class ZonedDateTimeDeserializer extends JsonDeserializer<ZonedDateTime> {&nbsp; &nbsp; @Override&nbsp; &nbsp; public ZonedDateTime deserialize(JsonParser p, DeserializationContext ctxt) throws IOException, JsonProcessingException {&nbsp; &nbsp; &nbsp; &nbsp; DateTimeFormatter dateTimeFormatter=DateTimeFormatter.ofPattern("dd-MM-yyyy HH:mm:ss z");&nbsp; &nbsp; &nbsp; &nbsp; LocalDate localDate = LocalDate.parse(p.getText(),dateTimeFormatter);&nbsp; &nbsp; &nbsp; &nbsp; return localDate.atStartOfDay(ZoneOffset.UTC);&nbsp; &nbsp; }}步骤02@JsonDeserialize在方法级别注释的支持下,您必须在 POJO 类中将反序列化类与受影响的字段一起使用。@JsonDeserialize(using = ZonedDateTimeDeserializer.class)private ZonedDateTime startDate;步骤03以 ZonedDateTimeDeserializer 类给出的上述格式将值作为字符串传递&nbsp; &nbsp; &nbsp; &nbsp;"startDate" : "09-03-2003 10:15:00 Europe/Paris"

梵蒂冈之花

遗憾的是,如果不将 POJO 的类型更改为 LocalDate,那将很困难。我能想到的最接近的解决方案是为杰克逊写一个习惯JsonDeserializer,这绝对不是那种事情的好习惯。

HUX布斯

您可以编写自己的反序列化器,如@cassiomolin答案中所示。但也有另一种选择。在堆栈跟踪中,我们有DeserializationContext.weirdStringException方法允许我们提供DeserializationProblemHandler处理奇怪字符串值的方法。请参阅以下示例:import com.fasterxml.jackson.annotation.JsonFormat;import com.fasterxml.jackson.annotation.JsonFormat.Shape;import com.fasterxml.jackson.annotation.JsonProperty;import com.fasterxml.jackson.databind.DeserializationContext;import com.fasterxml.jackson.databind.ObjectMapper;import com.fasterxml.jackson.databind.deser.DeserializationProblemHandler;import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule;import java.io.IOException;import java.time.LocalDate;import java.time.ZonedDateTime;import java.time.format.DateTimeFormatter;import java.util.TimeZone;public class AppJson {&nbsp; &nbsp; public static void main(String[] args) throws IOException {&nbsp; &nbsp; &nbsp; &nbsp; ObjectMapper mapper = new ObjectMapper();&nbsp; &nbsp; &nbsp; &nbsp; // override default time zone if needed&nbsp; &nbsp; &nbsp; &nbsp; mapper.setTimeZone(TimeZone.getTimeZone("America/Los_Angeles"));&nbsp; &nbsp; &nbsp; &nbsp; mapper.registerModule(new JavaTimeModule());&nbsp; &nbsp; &nbsp; &nbsp; mapper.addHandler(new DeserializationProblemHandler() {&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; @Override&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; public Object handleWeirdStringValue(DeserializationContext ctxt, Class<?> targetType,&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; String valueToConvert, String failureMsg) {&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; LocalDate date = LocalDate.parse(valueToConvert, DateTimeFormatter.ISO_DATE);&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; return date.atStartOfDay(ctxt.getTimeZone().toZoneId());&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; }&nbsp; &nbsp; &nbsp; &nbsp; });&nbsp; &nbsp; &nbsp; &nbsp; String json = "{\"startDate\": \"2019-07-25\"}";&nbsp; &nbsp; &nbsp; &nbsp; Markdown markdown = mapper.readValue(json, Markdown.class);&nbsp; &nbsp; &nbsp; &nbsp; System.out.println(markdown);&nbsp; &nbsp; }}上面的代码打印:Markdown{startDate=2019-07-25T00:00-07:00[America/Los_Angeles]}
随时随地看视频慕课网APP

相关分类

Java
我要回答