Вопрос по json, java – Обработка альтернативных имен свойств в десериализации Джексона

11

Я работаю над преобразованием службы REST / JSON из Coldfusion 9 в приложение Spring-MVC 3.1. Я использую Jackson (1.9.5) и MappingJacksonJsonConverter, которые предоставляет Spring, и настраиваю ObjectMapper для именования полей с помощью CAMEL_CASE_TO_LOWER_CASE_WITH_UNDERSCORES.

Проблема, с которой я сталкиваюсь, заключается в том, что наша традиционная служба создает «верблюжий кейс для ВЕРХНЕГО кейса с подчеркиванием». как имена свойств JSON. Потребители этого JSON, также написанного на ColdFusion, могут меньше заботиться о случае, но Джексон действительно заботится о случае и выдает исключения UnrecognizedPropertyException.

Изучив практически все настройки, которые я мог получить из ObjectMapper - DeserializationConfig, DeserializerProvider и т. Д., Я получил очень грязный хак, в котором я анализировал дерево JSON, выводил его с помощью специального JsonGenerator, который в нижнем регистре имен полей, и затем анализирует его как объект.

<code>MappingJacksonHttpMessageConverter mc = new MappingJacksonHttpMessageConverter() {
    @Override
    protected Object readInternal(Class<?> clazz, HttpInputMessage inputMessage) throws IOException, HttpMessageNotReadableException {
        return this.getObjectMapper().readValue(translateToLowerCaseKeys(inputMessage.getBody()), getJavaType(clazz));
    }

    private byte[] translateToLowerCaseKeys(InputStream messageBody) throws IOException {
        StringWriter sw = new StringWriter();
        JsonGenerator lowerCaseFieldNameGenerator = new JsonGeneratorDelegate(this.getObjectMapper().getJsonFactory().createJsonGenerator(sw)) {
            @Override
            public void writeFieldName(String name) throws IOException, org.codehaus.jackson.JsonGenerationException {
                delegate.writeFieldName(name.toLowerCase());
            };
        };
        this.getObjectMapper().writeTree(lowerCaseFieldNameGenerator, this.getObjectMapper().readTree(messageBody));
        lowerCaseFieldNameGenerator.close();
        return sw.getBuffer().toString().getBytes();
    }
};
</code>

Это решение кажется очень неэффективным. Eстьрешение это работает для ключей карты, но я не смог найти аналогичное решение для имен полей.

Альтернативное решение состоит в том, чтобы иметь два установщика, один аннотированный устаревшим именем поля. Стратегия именования должна быть расширена, чтобы игнорировать эти поля, что в моей ситуации хорошо, поскольку средство отображения объектов не будет иметь дело с любыми другими классами со стратегией UPPER_UNDERSCORE:

<code>public class JsonNamingTest {
    public static class CaseInsensitive extends LowerCaseWithUnderscoresStrategy {
        public String translate(String in) {
            return (in.toUpperCase().equals(in) ? in : super.translate(in));
        }
    }

    public static class A {
        private String testField;
        public String getTestField() {
            return testField;
        }
        public void setTestField(String field) {
            this.testField = field;
        }
        @JsonProperty("TEST_FIELD")
        public void setFieldAlternate(String field) {
            this.testField = field;
        }
    }

    @Test
    public void something() throws Exception {
        A test = new A();
        test.setTestField("test");
        ObjectMapper mapper = new ObjectMapper().setPropertyNamingStrategy(new CaseInsensitive());
        assertEquals("{\"test_field\":\"test\"}", mapper.writeValueAsString(test));
        assertEquals("test", mapper.readValue("{\"test_field\":\"test\"}", A.class).getTestField());
        assertEquals("test", mapper.readValue("{\"TEST_FIELD\":\"test\"}", A.class).getTestField());
    }
}
</code>

Это более идеально, чем предыдущее решение, но для каждого поля потребуется два аннотированных установщика - один для нового формата, другой для поддержки устаревшего формата.

Кто-нибудь сталкивался со способом сделать Джексона нечувствительным к регистру для десериализации имен полей или в этом случае принять несколько псевдонимов для имени поля?

Ваш Ответ

4   ответа
12

ObjectMapper mapper = new ObjectMapper();
mapper.configure(MapperFeature.ACCEPT_CASE_INSENSITIVE_PROPERTIES, true);
mapper.enable(MapperFeature.ACCEPT_CASE_INSENSITIVE_PROPERTIES);Error: User Rate Limit Exceeded
1

Jackson2ObjectMapperBuilderCustomizer который сохраняет автоконфигурацию по умолчанию. Это можно сделать, добавив компонент:

@Bean
public Jackson2ObjectMapperBuilderCustomizer acceptCaseInsensitive() {
    return builder -> builder.featuresToEnable(MapperFeature.ACCEPT_CASE_INSENSITIVE_PROPERTIES);
}
5

а не Jackson 2.x, вы можете использовать MixIns. Из того, что я понимаю, вы хотите, чтобы десериализация понимала UPPER_CASE, но вы хотите сериализовать в нижнем регистре. Добавьте MixIn в конфигурацию десериализации, но не в конфигурацию сериализации:

public class JsonNamingTest {   
    public static class A {
        private String testField;
        public String getTestField() {
            return testField;
        }
        public void setTestField(String field) {
            this.testField = field;
        }
    }

    public static class MixInA {
        @JsonProperty("TEST_FIELD")
        public void setTestField(String field) {}
    }

    @Test
    public void something() throws Exception {
        A test = new A();
        test.setTestField("test");
        ObjectMapper mapper = new ObjectMapper();
        mapper.getDeserializationConfig().addMixInAnnotations(A.class, MixInA.class);
        assertEquals("{\"test_field\":\"test\"}", mapper.writeValueAsString(test));
        assertEquals("test", mapper.readValue("{\"test_field\":\"test\"}", A.class).getTestField());
        assertEquals("test", mapper.readValue("{\"TEST_FIELD\":\"test\"}", A.class).getTestField());
    }
}

Однако вы не можете сделать это с помощью Jackson 2.x: mixIns совместно используются конфигурацией сериализации и десериализации. Мне еще предстоит найти способ обработки псевдонимов с помощью Jackson 2.x :(

6

  class CaseInsensitiveNaming extends PropertyNamingStrategy {
    @Override
    public String nameForGetterMethod(MapperConfig<?> config,
         AnnotatedMethod method, String defaultName)
    {
      // case-insensitive, underscores etc. mapping to property names
      // so you need to implement the convert method in way you need.
      return convert(defaultName); 
    }
  }

  objectMapper.setPropertyNamingStrategy(new CaseInsensitiveNaming());
Error: User Rate Limit Exceeded Dan Watt
Error: User Rate Limit Exceeded Dan Watt

Похожие вопросы