Подтвердить что ты не робот

JSON Java 8 Формат LocalDateTime в Spring Загрузка

У меня возникла небольшая проблема с форматированием Java 8 LocalDateTime в моем загрузочном приложении Spring. С "нормальными" датами у меня нет проблем, но поля 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;
}

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

4b9b3361

Ответ 1

обновление: 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

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

Ответ 2

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

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

который не то, что я хотел, но я приближался. Затем я добавил следующее

spring.jackson.serialization.write_dates_as_timestamps=false

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

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

Ответ 3

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

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

Ответ 4

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 Конфигурация.

@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;
    }
}

Ответ 5

Я нашел другое решение, которое вы можете преобразовать в любой формат и применить ко всем типам данных 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

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

Я объединил несколько ответов здесь, и в итоге мой работал с чем-то вроде этого. (Я использую 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;

}

Ответ 7

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

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

<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());

Ответ 8

Как уже упоминалось, 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]));
    }

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

Ответ 9

Я использую 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;
    }

Ответ 10

@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();
    }
}

Ответ 11

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

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

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