使用 ObjectMapper + JavaTimeModule 注册 JacksonJsonProvider 到 Jersey 2 Client

2022-09-03 05:53:47

我正在尝试封送包含ISO格式时间戳的响应,如下所示:

{
...
    "time" : "2014-07-02T04:00:00.000000Z"
...
}

在我的领域模型对象中放入字段。最终,如果我使用以下代码段中注释的解决方案,它将起作用。关于SO有很多类似的问题,但我想得到一个具体的答案,另一种使用的方法有什么问题?ZonedDateTimeJacksonJsonProviderObjectMapper + JavaTimeModule

        ObjectMapper mapper = new ObjectMapper();
        mapper.registerModule(new JavaTimeModule());
        JacksonJsonProvider provider = new JacksonJsonProvider(mapper);
        Client client = ClientBuilder.newBuilder()
//                .register(new ObjectMapperContextResolver(){
//                    @Override
//                    public ObjectMapper getContext(Class<?> type) {
//                        ObjectMapper mapper = new ObjectMapper();
//                        mapper.registerModule(new JavaTimeModule());
//                        return mapper;
//                    }
//                })
                .register(provider)
                .register(JacksonFeature.class)
                .build();

我得到的错误:

javax.ws.rs.client.ResponseProcessingException: com.fasterxml.jackson.databind.JsonMappingException: Can not construct instance of java.time.ZonedDateTime: no String-argument constructor/factory method to deserialize from String value ('2017-02-24T20:46:05.000000Z')
 at [Source: org.glassfish.jersey.message.internal.ReaderInterceptorExecutor$UnCloseableInputStream@53941c2f

项目依赖关系包括:

compile 'com.fasterxml.jackson.datatype:jackson-datatype-jdk8:2.8.7'
compile 'com.fasterxml.jackson.datatype:jackson-datatype-jsr310:2.8.7'
compile 'com.fasterxml.jackson.core:jackson-core:2.8.7'
compile 'com.fasterxml.jackson.core:jackson-databind:2.8.7'
compile 'com.fasterxml.jackson.core:jackson-annotations:2.8.7'
compile 'org.glassfish.jersey.core:jersey-client:2.25.1'

编辑

反序列化发生在这里:

CandlesResponse<BidAskCandle> candlesResponse = webTarget.request()
                .header(HttpHeaders.AUTHORIZATION,"Bearer "+token)
                .accept(MediaType.APPLICATION_JSON)
                .get(new GenericType<CandlesResponse<BidAskCandle>>(){});

答案 1

最终,如果我使用以下代码段中注释的解决方案,它将起作用。

首先,您在列表中缺少一个依赖项,您也具有依赖项,这就是问题所在。

jersey-media-json-jackson

此模块依赖于具有 .当您注册 (随附的 ) 时,它会注册自己的 ,这似乎优先于您提供的任何内容。JacksonJsonProviderJacksonFeaturejersey-media-json-jacksonJacksonJaxbJsonProvider

当您使用 时,实际查找并使用它来解决 .这就是它起作用的原因。无论你使用还是注册了自己的(没有为它配置),都会起作用。ContextResolverJacksonJsonProviderContextResolverObjectMapperJacksonFeatureJacksonJsonProviderObjectMapperContextResovler

关于该模块的另一件事是,它参与了泽西岛的自动可发现机制,该机制注册了它。因此,即使您没有明确注册它,它仍然会被注册。避免注册的唯一方法是:jersey-media-json-jacksonJacksonFeature

  1. 禁用自动发现(如上一个链接中所述)1

  2. 不要使用 .只需使用杰克逊原生模块。不过,关于这一点的事情是,在本机模块之上添加了一些功能,因此您将丢失这些功能。jersey-media-json-jacksonjackson-jaxrs-json-providerjersey-media-json-jackson

  3. 尚未测试,但似乎如果您使用而不是 ,它可能会起作用。如果您查看JacksonFeature的来源,您将看到它检查是否已注册。如果有,它不会注册它自己的。JacksonJaxbJsonProviderJacksonJsonProviderJacksonJaxbJsonProvider

    我不确定的一件事是自动发现的。注册它的顺序,如果它会影响它是否捕获您的注册。您可以测试的东西。JacksonJaxbJsonProvider


脚注

1. 另请参见


答案 2

来自我的宠物项目:

<dependency>
    <groupId>com.fasterxml.jackson.datatype</groupId>
    <artifactId>jackson-datatype-jsr310</artifactId>
    <version>${jackson.version}</version>
</dependency>

public WebTarget getTarget(URI uri) {
    Client client = ClientBuilder
            .newClient()
            .register(JacksonConfig.class);
    return client.target(uri);
}

哪里

@Provider
public class JacksonConfig implements ContextResolver<ObjectMapper> {

    private final ObjectMapper objectMapper;

    public JacksonConfig() {
        objectMapper = new ObjectMapper();
        objectMapper.registerModule(new JavaTimeModule());
        objectMapper.disable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS);
        objectMapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
        objectMapper.setSerializationInclusion(JsonInclude.Include.NON_NULL);
    }

    @Override
    public ObjectMapper getContext(Class<?> aClass) {
        return objectMapper;
    }
}

推荐