反序列化Spring类时如何配置Spring使用的默认ObjectMapper以使用自定义反序列化器

时间:2019-06-23 10:05:50

标签: spring spring-boot spring-oauth2

每当我具有OAuth2AccessToken类型的类时,我都想在Spring的默认ObjectMapper中使用自己的自定义反序列化器。界面用

注释
JsonDeserialize(using = OAuth2AccessTokenJackson2Deserializer.class)

这是目前反序列化的功能,但我想使用自己的序列。

到目前为止,我已经创建了自己的自定义解串器

public class MyCustomDeserializer extends StdDeserializer<OAuth2AccessToken> {

public MyCustomDeserializer() {
    super(OAuth2AccessToken.class);
}

@Override
public OAuth2AccessToken deserialize(JsonParser jp, DeserializationContext ctxt) throws IOException,
        JsonProcessingException {

    String tokenValue = null;
    String tokenType = null;
    String refreshToken = null;
    Long expiresIn = null;
    Set<String> scope = null;
    Map<String, Object> additionalInformation = new LinkedHashMap<String, Object>();

    // TODO What should occur if a parameter exists twice
    while (jp.nextToken() != JsonToken.END_OBJECT) {
        String name = jp.getCurrentName();
        jp.nextToken();
        if (OAuth2AccessToken.ACCESS_TOKEN.equals(name)) {
            tokenValue = jp.getText();
        }
        else if (OAuth2AccessToken.TOKEN_TYPE.equals(name)) {
            tokenType = jp.getText();
        }
        else if (OAuth2AccessToken.REFRESH_TOKEN.equals(name)) {
            refreshToken = jp.getText();
        }
        else if (OAuth2AccessToken.EXPIRES_IN.equals(name)) {
            try {
                expiresIn = jp.getLongValue();
            } catch (JsonParseException e) {
                expiresIn = Long.valueOf(jp.getText());
            }
        }
        else if (OAuth2AccessToken.SCOPE.equals(name)) {
            scope = parseScope(jp);
        } else {
            additionalInformation.put(name, jp.readValueAs(Object.class));
        }
    }

    // TODO What should occur if a required parameter (tokenValue or tokenType) is missing?

    DefaultOAuth2AccessToken accessToken = new DefaultOAuth2AccessToken(tokenValue);
    accessToken.setTokenType(tokenType);
    if (expiresIn != null) {
        accessToken.setExpiration(new Date(System.currentTimeMillis() + (expiresIn * 1000)));
    }
    if (refreshToken != null) {
        accessToken.setRefreshToken(new DefaultOAuth2RefreshToken(refreshToken));
    }
    accessToken.setScope(scope);
    accessToken.setAdditionalInformation(additionalInformation);

    return accessToken;
}

private Set<String> parseScope(JsonParser jp) throws JsonParseException, IOException {
    Set<String> scope;
    if (jp.getCurrentToken() == JsonToken.START_ARRAY) {
        scope = new TreeSet<String>();
        while (jp.nextToken() != JsonToken.END_ARRAY) {
            scope.add(jp.getValueAsString());
        }
    } else {
        String text = jp.getText();
        scope = OAuth2Utils.parseParameterList(text);
    }
    return scope;
}

}

通过扩展DefaultOAuth2AccessToken我自己的自定义类

@com.fasterxml.jackson.databind.annotation.JsonDeserialize(using = MyCustomDeserializer.class)
public class MyCustomOAuth2AccessToken extends DefaultOAuth2AccessToken {

public MyCustomOAuth2AccessToken(String value) {
    super(value);
}

public MyCustomOAuth2AccessToken(OAuth2AccessToken accessToken) {
    super(accessToken);
}
}

此刻,我正在像这样注册一个Jackson2ObjectMapperBuilderCustomizer类型的bean

    @Bean
public Jackson2ObjectMapperBuilderCustomizer addCustomDeserialization() {
    return new Jackson2ObjectMapperBuilderCustomizer() {

        @Override
        public void customize(Jackson2ObjectMapperBuilder jacksonObjectMapperBuilder) {
            SimpleModule m = new SimpleModule();
            m.addDeserializer(OAuth2AccessToken.class, new MyCustomDeserializer());
            jacksonObjectMapperBuilder.modules(m);
        }
    };
}

    @Bean
public OAuth2ClientContext getOAuth2ClientContext() {
    DefaultOAuth2ClientContext defaultOAuth2ClientContext = new DefaultOAuth2ClientContext();
    defaultOAuth2ClientContext.setAccessToken(new MyCustomOAuth2AccessToken("test"));
    return defaultOAuth2ClientContext;
}

2 个答案:

答案 0 :(得分:0)

您可以简单地使用 @JsonComponent 注释反序列化类。 注释使我们可以将带注释的类公开为Jackson序列化器和/或反序列化器,而无需手动将其添加到ObjectMapper。

答案 1 :(得分:0)

要全局配置ObjectMapper,只需创建一个Jackson2ObjectMapperBuilder类型的bean并使用deserializerByType方法:

@Bean
public Jackson2ObjectMapperBuilder objectMapperBuilder() {
  return new Jackson2ObjectMapperBuilder()
           .deserializerByType(OAuth2AccessToken.class, new MyCustomDeserializer());
}

可以在here中找到在SpringBoot中配置ObjectMapper的参考。