我如何反序列化以杰克逊为单位的时间戳?
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;
就是这样
【讨论】:
以上是关于我如何反序列化以杰克逊为单位的时间戳?的主要内容,如果未能解决你的问题,请参考以下文章