我如何反序列化以杰克逊为单位的时间戳?

Posted

技术标签:

【中文标题】我如何反序列化以杰克逊为单位的时间戳?【英文标题】:How do I deserialize timestamps that are in seconds with Jackson? 【发布时间】:2014-01-05 07:21:08 【问题描述】:

我有一些以秒为单位的时间戳(即 Unix 时间戳)的 JSON:

"foo":"bar","timestamp":1386280997

要求 Jackson 将其反序列化为带有 DateTime 字段的对象以获取时间戳,结果为 1970-01-17T01:11:25.983Z,这是纪元后不久的时间,因为 Jackson 假设它以 毫秒 为单位。除了撕开 JSON 并添加一些零之外,我如何让 Jackson 理解 seconds 时间戳?

【问题讨论】:

有一个问题here展示了如何实现自定义日期反序列化类 感谢您的提示——我回答了这个问题,并详细说明了我是如何解决这个问题的,因为解串器与那个问题有点不同。 【参考方案1】:

我写了一个自定义的deserializer 以秒为单位处理时间戳(Groovy 语法)。

class UnixTimestampDeserializer extends JsonDeserializer<DateTime> 
    Logger logger = LoggerFactory.getLogger(UnixTimestampDeserializer.class)

    @Override
    DateTime deserialize(JsonParser jp, DeserializationContext ctxt) throws IOException, JsonProcessingException 
        String timestamp = jp.getText().trim()

        try 
            return new DateTime(Long.valueOf(timestamp + '000'))
         catch (NumberFormatException e) 
            logger.warn('Unable to deserialize timestamp: ' + timestamp, e)
            return null
        
    

然后我注释了我的 POGO 以将其用作时间戳:

class TimestampThing 
    @JsonDeserialize(using = UnixTimestampDeserializer.class)
    DateTime timestamp

    @JsonCreator
    public TimestampThing(@JsonProperty('timestamp') DateTime timestamp) 
        this.timestamp = timestamp
    

【讨论】:

你好,只是出于好奇,它是什么Java方言?谢谢! 在普通无聊的 java 中也能像魅力一样工作。只需在您需要的任何日期时间字段上使用注释。没有尝试使用 Java 8 Date,但那里也应该没有问题。谢谢 如果您可以使用 java8,您可能对使用 java.time.Instant 类和使用常规 jackson datatypes 感兴趣。这样你只需要@JsonProperty("timestamp") private Instant timestamp @Pierrick 请把它变成一个答案,这样我就可以投票了 ;-)【参考方案2】:
@JsonFormat(shape=JsonFormat.Shape.STRING, pattern="s")
public Date timestamp;

编辑:vivek-kothari 建议

@JsonFormat(shape=JsonFormat.Shape.NUMBER, pattern="s")
public Timestamp timestamp;

【讨论】:

为什么要投反对票?因为它比其他解决方案更容易? 可能是因为它的时间戳在 json 中表示为 integer 而不是 string jackson 确实需要类型转换 这行不通,因为它 s 不是指自 1970 年以来的秒数,而是指“给定”日期的秒数 (0..59) 这仅适用于 java.sql.Timestamp,如果您的 json 包含数字,则无法使其与 java.util.Date 一起使用【参考方案3】:

与 @DrewStephens 的方法非常相似,它使用 Java SE TimeUnit API(在 JDK1.5 中引入)而不是普通的字符串连接,因此(可以说)更简洁,更具表现力:

public class UnixTimestampDeserializer extends JsonDeserializer<Date> 

    @Override
    public Date deserialize(JsonParser parser, DeserializationContext context) 
            throws IOException, JsonProcessingException 
        String unixTimestamp = parser.getText().trim();
        return new Date(TimeUnit.SECONDS.toMillis(Long.valueOf(unixTimestamp)));
    

在受影响的Date 字段上指定您的自定义反序列化器 (UnixTimestampDeserializer):

@JsonDeserialize(using = UnixTimestampDeserializer.class)
private Date updatedAt;

【讨论】:

【参考方案4】:

我遇到了这个确切的问题,除了我的 ZonedDateTime 对象变成了 unix-timestamps(秒)并且我需要它们以毫秒为单位(在浏览器上初始化 JS Date 对象)。

实现一个自定义的序列化器/反序列化器对于一些应该非常简单的事情来说看起来工作量太大,所以我在别处寻找,发现我可以配置对象映射器以获得所需的结果。

因为我的应用程序已经覆盖了 Jersey 提供的默认 ObjectMapper,所以我只需要禁用 SerializationFeature.WRITE_DATE_TIMESTAMPS_AS_NANOSECONDS

这是我的代码

@Provider
public class RPObjectMapperProvider implements ContextResolver<ObjectMapper> 

    final ObjectMapper defaultObjectMapper;

    public RPObjectMapperProvider() 
        defaultObjectMapper = new ObjectMapper();

        // this turned ZonedDateTime objects to proper seconds timestamps 
        defaultObjectMapper.enable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS);

        // disable to serialize dates for millis timestamps
        defaultObjectMapper.disable(SerializationFeature.WRITE_DATE_TIMESTAMPS_AS_NANOSECONDS);

        // using Java8 date time classes
        defaultObjectMapper.registerModule(new JavaTimeModule());
    

    @Override
    public ObjectMapper getContext(Class<?> type) 
        return defaultObjectMapper;
    

就是这样

【讨论】:

以上是关于我如何反序列化以杰克逊为单位的时间戳?的主要内容,如果未能解决你的问题,请参考以下文章

如何使用杰克逊制作自定义反序列化器将数字(即部门 ID)转换为部门对象?

为日期反序列化设置杰克逊时区

有啥方法可以防止杰克逊中的字段反序列化?

杰克逊:地图的反序列化

杰克逊反序列化阵列通量

反序列化的改造/杰克逊错误