it-swarm.com.ru

JSON Java 8 LocalDateTime формат в Spring Boot

У меня небольшая проблема с форматированием Java 8 LocalDateTime в моем приложении Spring Boot. С «нормальными» датами у меня нет проблем, но поля LocalDateTime преобразуются в следующее:

"startDate" : {
    "year" : 2010,
    "month" : "JANUARY",
    "dayOfMonth" : 1,
    "dayOfWeek" : "FRIDAY",
    "dayOfYear" : 1,
    "monthValue" : 1,
    "hour" : 2,
    "minute" : 2,
    "second" : 0,
    "nano" : 0,
    "chronology" : {
      "id" : "ISO",
      "calendarType" : "iso8601"
    }
  }

Хотя я хотел бы преобразовать его в нечто вроде:

"startDate": "2015-01-01"

Мой код выглядит так:

@JsonFormat(pattern="yyyy-MM-dd")
@DateTimeFormat(iso = DateTimeFormat.ISO.TIME)
public LocalDateTime getStartDate() {
    return startDate;
}

Но ни одна из вышеупомянутых аннотаций не работает, дата продолжает форматироваться, как указано выше. Предложения приветствуются!

84
Erik Pragt

update: Spring Boot 2.x больше не требует этой конфигурации. Я написал более актуальный ответ здесь .


(Это способ сделать это до Spring Boot 2.x, это может быть полезно для людей, работающих над старой версией Spring Boot)

Я наконец нашел здесь как это сделать. Чтобы это исправить, мне нужна была другая зависимость:

compile("com.fasterxml.jackson.datatype:jackson-datatype-jsr310:2.4.0")

Включив эту зависимость, Spring автоматически зарегистрирует конвертер для нее, как описано здесь . После этого вам нужно добавить следующее в application.properties:

spring.jackson.serialization.write_dates_as_timestamps=false

Это обеспечит использование правильного конвертера, а даты будут напечатаны в формате 2016-03-16T13:56:39.492

Аннотации нужны только в том случае, если вы хотите изменить формат даты.

108
Erik Pragt

Я добавил com.fasterxml.jackson.datatype: jackson-datatype-jsr310: 2.6.1 dependency и начал получать дату в следующем формате:

"birthDate": [
    2016,
    1,
    25,
    21,
    34,
    55
  ]

это не то, что я хотел, но я становился ближе. Затем я добавил следующее

spring.jackson.serialization.write_dates_as_timestamps=false

в файл application.properties, который дал мне правильный формат, который мне был нужен.

"birthDate": "2016-01-25T21:34:55"
77
patelb

Вот он в Maven, со свойством, чтобы вы могли выжить между весенними обновлениями загрузки

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

1) Зависимость

 compile group: 'com.fasterxml.jackson.datatype', name: 'jackson-datatype-jsr310', version: '2.8.8' 

2) Аннотация с форматом даты и времени.

public class RestObject {

    private LocalDateTime timestamp;

    @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
    public LocalDateTime getTimestamp() {
        return timestamp;
    }
}

3) Spring Config.

@Configuration
public class JacksonConfig {

    @Bean
    @Primary
    public ObjectMapper objectMapper(Jackson2ObjectMapperBuilder builder) {
        System.out.println("Config is starting.");
        ObjectMapper objectMapper = builder.createXmlMapper(false).build();
        objectMapper.configure(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS, false);
        return objectMapper;
    }
}
10
Yan Khonski

Я нашел другое решение, которое вы можете преобразовать в любой формат и применить ко всем типам данных LocalDateTime, и вам не нужно указывать @JsonFormat над каждым типом данных LocalDateTime . Сначала добавьте зависимость: 

<dependency>
    <groupId>com.fasterxml.jackson.datatype</groupId>
    <artifactId>jackson-datatype-jsr310</artifactId>
</dependency>

Добавьте следующий боб: 

@Configuration
public class Java8DateTimeConfiguration {
    /**
     * Customizing
     * http://docs.spring.io/spring-boot/docs/current/reference/html/howto-spring-mvc.html
     *
     * Defining a @Bean of type Jackson2ObjectMapperBuilder will allow you to customize both default ObjectMapper and XmlMapper (used in MappingJackson2HttpMessageConverter and MappingJackson2XmlHttpMessageConverter respectively).
     */
    @Bean
    public Module jsonMapperJava8DateTimeModule() {
        val bean = new SimpleModule();

        bean.addDeserializer (ZonedDateTime.class, new JsonDeserializer<ZonedDateTime>() {
            @Override
            public ZonedDateTime deserialize(JsonParser jsonParser, DeserializationContext deserializationContext) throws IOException, JsonProcessingException {
                return ZonedDateTime.parse(jsonParser.getValueAsString(), DateTimeFormatter.ISO_ZONED_DATE_TIME);
            }
        });

        bean.addDeserializer(LocalDateTime.class, new JsonDeserializer<LocalDateTime>() {
            @Override
            public LocalDateTime deserialize(JsonParser jsonParser, DeserializationContext deserializationContext) throws IOException, JsonProcessingException {
                return LocalDateTime.parse(jsonParser.getValueAsString(), DateTimeFormatter.ISO_LOCAL_DATE_TIME);
            }
        });

        bean.addSerializer(ZonedDateTime.class, new JsonSerializer<ZonedDateTime>() {
            @Override
            public void serialize(
                    ZonedDateTime zonedDateTime, JsonGenerator jsonGenerator, SerializerProvider serializerProvider)
                    throws IOException {
                jsonGenerator.writeString(DateTimeFormatter.ISO_ZONED_DATE_TIME.format(zonedDateTime));
            }
        });

        bean.addSerializer(LocalDateTime.class, new JsonSerializer<LocalDateTime>() {
            @Override
            public void serialize(
                    LocalDateTime localDateTime, JsonGenerator jsonGenerator, SerializerProvider serializerProvider)
                    throws IOException {
                jsonGenerator.writeString(DateTimeFormatter.ISO_LOCAL_DATE_TIME.format(localDateTime));
            }
        });

        return bean;
    }
}

в вашем конфигурационном файле добавьте следующее: 

@Import(Java8DateTimeConfiguration.class)

Это будет сериализовать и десериализовать все свойства LocalDateTime и ZonedDateTime, если вы используете objectMapper, созданный Spring.

Формат, который вы получили для ZonedDateTime: «2017-12-27T08: 55: 17.317 + 02: 00 [Азия/Иерусалим]» Для LocalDateTime: «2017-12-27T09: 05: 30.523»

6
Ida Amit

Это прекрасно работает:

Добавьте зависимость:

<dependency>
    <groupId>com.fasterxml.jackson.datatype</groupId>
    <artifactId>jackson-datatype-jdk8</artifactId>
</dependency>

Добавьте аннотацию:

@JsonFormat(pattern="yyyy-MM-dd")

Теперь вы должны получить правильный формат.

Чтобы использовать объект маппер, вам нужно зарегистрировать JavaTime

ObjectMapper mapper = new ObjectMapper();
mapper.registerModule(new JavaTimeModule());
2
Manuel Velasquez

Пишу этот ответ как напоминание и для меня.

Я объединил несколько ответов здесь, и в конце концов мой работал с чем-то вроде этого ... (. Я использую SpringBoot 1.5.7 и Lombok 1.16.16)

@Data
public Class someClass {

   @DateTimeFormat(iso = DateTimeFormat.ISO.DATE_TIME)
   @JsonSerialize(using = LocalDateTimeSerializer.class)
   @JsonDeserialize(using = LocalDateTimeDeserializer.class)
   private LocalDateTime someDate;

}
2
JWiryo

Как уже упоминалось, Spring-Boot будет загружать все, что вам нужно (как для веб-, так и для начала webflux).

Но что еще лучше - вам не нужно регистрировать какие-либо модули самостоятельно .. Посмотрите здесь . Поскольку @SpringBootApplication использует @EnableAutoConfiguration под капотом, это означает, что JacksonAutoConfiguration будет автоматически добавлен в контекст .. Теперь, если вы заглянете внутрь JacksonAutoConfiguration, вы увидите:

    private void configureModules(Jackson2ObjectMapperBuilder builder) {
        Collection<Module> moduleBeans = getBeans(this.applicationContext,
                Module.class);
        builder.modulesToInstall(moduleBeans.toArray(new Module[0]));
    }

Этот fella будет вызван в процессе инициализации и извлечет все модули, которые он может найти в пути к классам . (Я использую Spring Boot 2.1)

0
yuranos87

@JsonDeserialize(using= LocalDateDeserializer.class) не работает для меня с приведенной ниже зависимостью.

<dependency>
    <groupId>com.fasterxml.jackson.datatype</groupId>
    <artifactId>jackson-datatype-jsr310</artifactId>
    <version> 2.9.6</version>
</dependency>

Я использовал приведенный ниже преобразователь кода для десериализации даты в Java.sql.Date.

import javax.persistence.AttributeConverter;
import javax.persistence.Converter;


@SuppressWarnings("UnusedDeclaration")
@Converter(autoApply = true)
public class LocalDateConverter implements AttributeConverter<Java.time.LocalDate, Java.sql.Date> {


    @Override
    public Java.sql.Date convertToDatabaseColumn(Java.time.LocalDate attribute) {

        return attribute == null ? null : Java.sql.Date.valueOf(attribute);
    }

    @Override
    public Java.time.LocalDate convertToEntityAttribute(Java.sql.Date dbData) {

        return dbData == null ? null : dbData.toLocalDate();
    }
}
0
Ram

Я использую Springboot 2.0.6 и по какой-то причине изменения в приложении yml не работают. А также у меня было больше требований.

Я попытался создать ObjectMapper и пометить его как Primary, но весенняя загрузка жаловалась, что у меня уже есть jacksonObjectMapper как помеченный как Primary !!

Так вот что я сделал. Я внес изменения во внутренний картограф.

Мой Serializer и Deserializer особенные - они имеют дело с 'dd/MM/YYYY'; и при десериализации - он старается использовать 3-4 популярных формата, чтобы убедиться, что у меня есть LocalDate.

@Autowired
    ObjectMapper mapper;

    @PostConstruct
    public ObjectMapper configureMapper() {
        mapper.setSerializationInclusion(JsonInclude.Include.NON_NULL);
        mapper.enable(DeserializationFeature.ACCEPT_EMPTY_STRING_AS_NULL_OBJECT);

        mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
        mapper.configure(SerializationFeature.ORDER_MAP_ENTRIES_BY_KEYS, true);

        mapper.configure(MapperFeature.ALLOW_COERCION_OF_SCALARS, true);
        mapper.configure(MapperFeature.SORT_PROPERTIES_ALPHABETICALLY, true);

        SimpleModule module = new SimpleModule();
        module.addDeserializer(LocalDate.class, new LocalDateDeserializer());
        module.addSerializer(LocalDate.class, new LocalDateSerializer());
        mapper.registerModule(module);

        return mapper;
    }
0
Kalpesh Soni

просто используйте:

@JsonFormat(pattern="10/04/2019")

или вы можете использовать шаблон, как вам нравится, например: ('-' in place of '/')

0
Ashish Dwivedi