Налаштування ObjectMapper навесні


91

моя мета - налаштувати objectMapperтак, щоб він лише серіалізував елемент, анотований @JsonProperty.

Для цього я дотримувався цього пояснення, яке говорить про те, як налаштувати об'єктну карту.

Я включив спеціальну об'єктну карту, як описано тут .

Однак, коли клас NumbersOfNewEventsсеріалізований, він все ще містить усі атрибути в json.

Хтось має підказку? Спасибі заздалегідь

Джексон 1.8.0 весна 3.0.5

CustomObjectMapper

public class CompanyObjectMapper extends ObjectMapper {
    public CompanyObjectMapper() {
        super();
        setVisibilityChecker(getSerializationConfig()
                .getDefaultVisibilityChecker()
                .withCreatorVisibility(JsonAutoDetect.Visibility.NONE)
                .withFieldVisibility(JsonAutoDetect.Visibility.NONE)
                .withGetterVisibility(JsonAutoDetect.Visibility.NONE)
                .withIsGetterVisibility(JsonAutoDetect.Visibility.NONE)
                .withSetterVisibility(JsonAutoDetect.Visibility.DEFAULT));
    }
}

servlet.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:context="http://www.springframework.org/schema/context"
    xmlns:mvc="http://www.springframework.org/schema/mvc"
    xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.0.xsd
        http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-3.0.xsd">

    <context:component-scan base-package="de.Company.backend.web" />

    <mvc:annotation-driven />

    <bean
        class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter">
        <property name="messageConverters">
            <list>
                <bean class="org.springframework.http.converter.json.MappingJacksonHttpMessageConverter">
                    <property name="objectMapper" ref="jacksonObjectMapper" />
                </bean>
            </list>
        </property>
    </bean>

    <bean id="jacksonObjectMapper" class="de.Company.backend.web.CompanyObjectMapper" />
</beans>

NumbersOfNewEvents

public class NumbersOfNewEvents implements StatusAttribute {

    public Integer newAccepts;
    public Integer openRequests;

    public NumbersOfNewEvents() {
        super();
    }
}

Відповіді:


136

За допомогою Spring Boot (1.2.4) та Jackson (2.4.6) для мене працювала наступна конфігурація на основі анотацій.

@Configuration
public class JacksonConfiguration {

    @Bean
    public ObjectMapper objectMapper() {
        ObjectMapper mapper = new ObjectMapper();
        mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
        mapper.configure(MapperFeature.DEFAULT_VIEW_INCLUSION, true);

        return mapper;
    }
}

3
Spring підтримує налаштування деяких властивостей Джексона в application.properties: docs.spring.io/spring-boot/docs/current/reference/html/…
NangSaigon

10
Гей, хлопці. Я зіткнувся з проблемою, що коли ви оголошуєте bean з іменем objectMapper у конфігурації Java, як показано в прикладі, цей компонент ігнорується Spring Boot, тому що bean з таким іменем вже зареєстрований у системі, навіть якщо ви розміщуєте \ @ Первинна анотація на ньому. Щоб вирішити цю проблему, вам потрібно зареєструвати bean з іншим ім'ям, наприклад, jsonObjectMapper, і позначити його анотацією \ @Primary
Demwis

Зверніться до моєї відповіді, якщо ви не використовуєте Spring Boot.
Соня

Питання: ви повинні назвати клас JacksonConfiguration? Це автоматичний спосіб доручити Spring Boot використовувати цей клас як клас конфігурації для Джексона?
Марко Сулла

Чи справді це працює при використанні також @EnableWebMvc? Я сумніваюся, бо ... див. У моїй відповіді : Чому використання @EnableWebMvc є проблемою?
adrhc

25

Можливо, тому, що я використовую Spring 3.1 (замість Spring 3.0.5, як вказано у вашому запитанні), але відповідь Стіва Іствуда мені не спрацювала. Це рішення працює для Spring 3.1:

У вашому весняному контексті xml:

<mvc:annotation-driven>
    <mvc:message-converters>
        <bean class="org.springframework.http.converter.StringHttpMessageConverter"/>
        <bean class="org.springframework.http.converter.ByteArrayHttpMessageConverter"/>
        <bean class="org.springframework.http.converter.json.MappingJacksonHttpMessageConverter">
            <property name="objectMapper" ref="jacksonObjectMapper" />
        </bean>        
    </mvc:message-converters>
</mvc:annotation-driven>

<bean id="jacksonObjectMapper" class="de.Company.backend.web.CompanyObjectMapper" />

Це допомогло для використання Jackson 2.x з Spring 3.1: stackoverflow.com/questions/10420040/…
Арам Кочарян

Корисно! Мені довелося застосувати такий підхід до реєстрації мого користувацького ObjectMapper для оновлення з Spring 3.0 до 3.2. Раніше встановлював властивість objectMapper під час визначення компонента MappingJacksonJsonView, але це більше не спрацьовувало. Це з Jackson 1.7
David Easley

20

Є org.springframework.http.converter.json.Jackson2ObjectMapperFactoryBeanдавно. Починаючи з 1.2 випуску Spring Boot існує org.springframework.http.converter.json.Jackson2ObjectMapperBuilderJava Config.

У String Boot конфігурація може бути такою простою, як:

spring.jackson.deserialization.<feature_name>=true|false
spring.jackson.generator.<feature_name>=true|false
spring.jackson.mapper.<feature_name>=true|false
spring.jackson.parser.<feature_name>=true|false
spring.jackson.serialization.<feature_name>=true|false
spring.jackson.default-property-inclusion=always|non_null|non_absent|non_default|non_empty

в classpath:application.propertiesабо якийсь код Java у @Configurationкласі:

@Bean
public Jackson2ObjectMapperBuilder jacksonBuilder() {
    Jackson2ObjectMapperBuilder builder = new Jackson2ObjectMapperBuilder();
    builder.indentOutput(true).dateFormat(new SimpleDateFormat("yyyy-MM-dd"));
    return builder;
}

Подивитися:


2
Це працює для більшості випадків використання, однак конфігурація джексона Spring Boot ігнорується, якщо ви використовуєте анотацію @EnableWebMvc (перевірено в Spring Boot 2.0.3).
Джон

1
Хоча я не тестував, я впевнений, що це не буде працювати з @EnableWebMvc (незважаючи на використання чи не весняне завантаження). Чому? см WebMvcConfigurationSupport (DelegatingWebMvcConfiguration розширює WebMvcConfigurationSupport): Jackson2ObjectMapperBuilder builder = Jackson2ObjectMapperBuilder.json().
adrhc

16

Я використовував це з Jackson 2.x та Spring 3.1.2+

servlet-context.xml:

Зверніть увагу, що кореневим елементом є <beans:beans>, тому вам може знадобитися видалити beansта додати mvcдеякі з цих елементів залежно від налаштування.

    <annotation-driven>
        <message-converters>
            <beans:bean
                class="org.springframework.http.converter.StringHttpMessageConverter" />
            <beans:bean
                class="org.springframework.http.converter.ResourceHttpMessageConverter" />
            <beans:bean
                class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter">
                <beans:property name="objectMapper" ref="jacksonObjectMapper" />
            </beans:bean>
        </message-converters>
    </annotation-driven>

    <beans:bean id="jacksonObjectMapper"
        class="au.edu.unimelb.atcom.transfer.json.mappers.JSONMapper" />

au.edu.unimelb.atcom.transfer.json.mappers.JSONMapper.java:

public class JSONMapper extends ObjectMapper {

    public JSONMapper() {
        SimpleModule module = new SimpleModule("JSONModule", new Version(2, 0, 0, null, null, null));
        module.addSerializer(Date.class, new DateSerializer());
        module.addDeserializer(Date.class, new DateDeserializer());
        // Add more here ...
        registerModule(module);
    }

}

DateSerializer.java:

public class DateSerializer extends StdSerializer<Date> {

    public DateSerializer() {
        super(Date.class);
    }

    @Override
    public void serialize(Date date, JsonGenerator json,
            SerializerProvider provider) throws IOException,
            JsonGenerationException {
        // The client side will handle presentation, we just want it accurate
        DateFormat df = StdDateFormat.getBlueprintISO8601Format();
        String out = df.format(date);
        json.writeString(out);
    }

}

DateDeserializer.java:

public class DateDeserializer extends StdDeserializer<Date> {

    public DateDeserializer() {
        super(Date.class);
    }

    @Override
    public Date deserialize(JsonParser json, DeserializationContext context)
            throws IOException, JsonProcessingException {
        try {
            DateFormat df = StdDateFormat.getBlueprintISO8601Format();
            return df.parse(json.getText());
        } catch (ParseException e) {
            return null;
        }
    }

}

10

Зараз я знайшов рішення на основі https://github.com/FasterXML/jackson-module-hibernate

Я розширив відображення об'єктів та додав атрибути у успадкований конструктор.

Потім нове відображення об’єкта реєструється як компонент.

<!-- https://github.com/FasterXML/jackson-module-hibernate -->
<bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter">
    <property name="messageConverters">
        <array>
            <bean id="jsonConverter"
            class="org.springframework.http.converter.json.MappingJacksonHttpMessageConverter">
                <property name="objectMapper">
                    <bean class="de.company.backend.spring.PtxObjectMapper"/>
                </property>
            </bean>
        </array>
    </property>
</bean>   

Щиро дякую за рішення! Я витратив більше чотирьох годин, копаючись навколо <annotation-driven> , він просто не підібрав перетворювачі. Спробував відповісти, нарешті це працює!
snowindy

1
Щоб використовувати анотації, ви можете просто анотувати свій власний Mapper, який розширюється ObjectMapperяк @Componentі @Primary. Це підхід, рекомендований у Весняному довідковому посібнику, і він у мене спрацював добре.
Ольга Маціашек

9

Якщо ви хочете додати власний ObjectMapper для реєстрації користувацьких серіалізаторів, спробуйте мою відповідь.

У моєму випадку (Spring 3.2.4 та Jackson 2.3.1), конфігурація XML для користувацького серіалізатора:

<mvc:annotation-driven>
    <mvc:message-converters register-defaults="false">
        <bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter">
            <property name="objectMapper">
                <bean class="org.springframework.http.converter.json.Jackson2ObjectMapperFactoryBean">
                    <property name="serializers">
                        <array>
                            <bean class="com.example.business.serializer.json.CustomObjectSerializer"/>
                        </array>
                    </property>
                </bean>
            </property>
        </bean>
    </mvc:message-converters>
</mvc:annotation-driven>

було незрозумілим чином перезаписано дещо за замовчуванням.

Це працювало для мене:

CustomObject.java

@JsonSerialize(using = CustomObjectSerializer.class)
public class CustomObject {

    private Long value;

    public Long getValue() {
        return value;
    }

    public void setValue(Long value) {
        this.value = value;
    }
}

CustomObjectSerializer.java

public class CustomObjectSerializer extends JsonSerializer<CustomObject> {

    @Override
    public void serialize(CustomObject value, JsonGenerator jgen,
        SerializerProvider provider) throws IOException,JsonProcessingException {
        jgen.writeStartObject();
        jgen.writeNumberField("y", value.getValue());
        jgen.writeEndObject();
    }

    @Override
    public Class<CustomObject> handledType() {
        return CustomObject.class;
    }
}

У <mvc:message-converters>(...)</mvc:message-converters>моєму рішенні не потрібна конфігурація XML ( ).


6

Я використовую Spring 4.1.6 та Jackson FasterXML 2.1.4.

    <mvc:annotation-driven>
        <mvc:message-converters>
            <bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter">
                <property name="objectMapper">
                    <bean class="com.fasterxml.jackson.databind.ObjectMapper">
                        <!-- 设置不输出null字段-->
                        <property name="serializationInclusion" value="NON_NULL"/>
                    </bean>
                </property>
            </bean>
        </mvc:message-converters>
    </mvc:annotation-driven>

це працює в моїй конфіграції applicationContext.xml


6

РІШЕННЯ 1

Перше робоче рішення (тестоване) корисне, особливо при використанні @EnableWebMvc:

@Configuration
@EnableWebMvc
public class WebConfig implements WebMvcConfigurer {
    @Autowired
    private ObjectMapper objectMapper;// created elsewhere
    @Override
    public void extendMessageConverters(List<HttpMessageConverter<?>> converters) {
        // this won't add a 2nd MappingJackson2HttpMessageConverter 
        // as the SOLUTION 2 is doing but also might seem complicated
        converters.stream().filter(c -> c instanceof MappingJackson2HttpMessageConverter).forEach(c -> {
            // check default included objectMapper._registeredModuleTypes,
            // e.g. Jdk8Module, JavaTimeModule when creating the ObjectMapper
            // without Jackson2ObjectMapperBuilder
            ((MappingJackson2HttpMessageConverter) c).setObjectMapper(this.objectMapper);
        });
    }

РІШЕННЯ 2

Звичайно, загальний підхід нижче теж працює (також робота з @EnableWebMvc):

@Configuration
@EnableWebMvc
public class WebConfig implements WebMvcConfigurer {
    @Autowired
    private ObjectMapper objectMapper;// created elsewhere
    @Override
    public void extendMessageConverters(List<HttpMessageConverter<?>> converters) {
        // this will add a 2nd MappingJackson2HttpMessageConverter 
        // (additional to the default one) but will work and you 
        // won't lose the default converters as you'll do when overwriting
        // configureMessageConverters(List<HttpMessageConverter<?>> converters)
        // 
        // you still have to check default included
        // objectMapper._registeredModuleTypes, e.g.
        // Jdk8Module, JavaTimeModule when creating the ObjectMapper
        // without Jackson2ObjectMapperBuilder
        converters.add(new MappingJackson2HttpMessageConverter(this.objectMapper));
    }

Чому використання @EnableWebMvc є проблемою?

@EnableWebMvc використовує DelegatingWebMvcConfigurationрозширення, WebMvcConfigurationSupportщо робить це:

if (jackson2Present) {
    Jackson2ObjectMapperBuilder builder = Jackson2ObjectMapperBuilder.json();
    if (this.applicationContext != null) {
        builder.applicationContext(this.applicationContext);
    }
    messageConverters.add(new MappingJackson2HttpMessageConverter(builder.build()));
}

це означає, що неможливо ввести власну, ObjectMapperщоб підготувати її до використання за замовчуванням MappingJackson2HttpMessageConverterпри використанні @EnableWebMvc.


Чудово! Насправді RepositoryRestMvcConfiguration, частина spring-data-rest-webmvc: 3.2.3 не використовує ObjectMapper, створений Spring, і лише користувацькі objectMapper з модулями не працюють нестандартно. Перший підхід є фактичним і замінює ObjectMapper, налаштований spring-data-rest-webmvc. Дякую!
Вальтоні Боавентура,

У мене виникла проблема, через яку я не зміг налаштувати свій ObjectMapper, який був жалюгідним б / к EnableWebMvc. Так рада, що натрапила на це.
Еван Вайт

3

У Spring Boot 2.2.xвам потрібно налаштувати його так:

@Bean
public ObjectMapper objectMapper(Jackson2ObjectMapperBuilder builder) {
    return builder.build()
}

Котлін:

@Bean
fun objectMapper(builder: Jackson2ObjectMapperBuilder) = builder.build()

2

Над Spring 4 немає необхідності налаштовувати, MappingJacksonHttpMessageConverterякщо ви лише маєте намір налаштувати ObjectMapper.

(конфігурація MappingJacksonHttpMessageConverterпризведе до втрати іншого MessageConverter)

Вам просто потрібно зробити:

public class MyObjectMapper extends ObjectMapper {

    private static final long serialVersionUID = 4219938065516862637L;

    public MyObjectMapper() {
        super();
        enable(SerializationFeature.INDENT_OUTPUT);
    }       
}

І у вашій конфігурації Spring створіть цей компонент:

@Bean 
public MyObjectMapper myObjectMapper() {        
    return new MyObjectMapper();
}

Не працює - - customObjectMapper: визначено методом 'customObjectMapper' у ресурсі шляху до класу [com / Config.class] -_halObjectMapper: defined in null
Angshuman Agarwal

1
І це буде працювати лише з Spring Boot. Не звичайний Весняний Інтернет.
Соня

2

Щоб налаштувати перетворювач повідомлень у простому Spring-Web, у цьому випадку, щоб увімкнути Java 8 JSR-310 JavaTimeModule, спочатку потрібно реалізувати WebMvcConfigurerу своєму @Configurationкласі, а потім замінити configureMessageConvertersметод:

@Override
public void configureMessageConverters(List<HttpMessageConverter<?>> converters) {
    ObjectMapper objectMapper = Jackson2ObjectMapperBuilder.json().modules(new JavaTimeModule(), new Jdk8Module()).build()
            .configure(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS, false);
    converters.add(new MappingJackson2HttpMessageConverter(objectMapper));
}

Подібно до цього, ви можете зареєструвати будь-який призначений для користувача ObjectMapperв конфігурації Spring на основі Java.


Я хотів би, щоб ObjectMapper вводився та конфігурувався через Spring Boot.
Константино Кронембергер

1
Звичайно, якщо ви використовуєте Spring Boot. Це не було питання Spring Boot.
Соня

Так, цікаво помітити, що при використанні Spring Web із Spring Boot він не використовує ObjectMapper, наданий Spring Boot, і саме тому я опинився у цьому питанні, яке мені допомогло.
Constantino Cronemberger

Це очистить усі перетворювачі повідомлень за замовчуванням! Я майже впевнений, що цього ніхто не хоче. Дивіться мою відповідь на рішення.
adrhc

Ну, якщо ви хочете розширити перетворювачі повідомлень, правильним рішенням буде перевизначення extendMessageConverters, згідно з документацією Spring . Але у мене в цьому не було потреби, бо мені потрібна лише індивідуальна MappingJackson2HttpMessageConverter.
Соня

1

Я використовую Spring 3.2.4 та Jackson FasterXML 2.1.1.

Я створив власний JacksonObjectMapper, який працює з явними анотаціями для кожного атрибута відображених об'єктів:

package com.test;
import com.fasterxml.jackson.annotation.JsonAutoDetect;
import com.fasterxml.jackson.annotation.PropertyAccessor;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.SerializationFeature;

public class MyJaxbJacksonObjectMapper extends ObjectMapper {

public MyJaxbJacksonObjectMapper() {

    this.setVisibility(PropertyAccessor.FIELD, JsonAutoDetect.Visibility.ANY)
            .setVisibility(PropertyAccessor.CREATOR, JsonAutoDetect.Visibility.ANY)
            .setVisibility(PropertyAccessor.SETTER, JsonAutoDetect.Visibility.NONE)
            .setVisibility(PropertyAccessor.GETTER, JsonAutoDetect.Visibility.NONE)
            .setVisibility(PropertyAccessor.IS_GETTER, JsonAutoDetect.Visibility.NONE);

    this.configure(SerializationFeature.FAIL_ON_EMPTY_BEANS, false);
    }
}

Потім це створюється у контексті конфігурації ( servlet-context.xml ):

<mvc:annotation-driven>
    <mvc:message-converters>

        <beans:bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter">
            <beans:property name="objectMapper">
                <beans:bean class="com.test.MyJaxbJacksonObjectMapper" />
            </beans:property>
        </beans:bean>

    </mvc:message-converters>
</mvc:annotation-driven>

Це чудово працює!

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.