[java] Java 8 LocalDate Jackson 형식

들어 java.util.Date 내가 할 때

@JsonFormat(shape = JsonFormat.Shape.STRING, pattern = "dd/MM/yyyy")
  private Date dateOfBirth;

그런 다음 JSON 요청에서 보낼 때

{ {"dateOfBirth":"01/01/2000"} }  

효과가있다.

Java 8의 LocalDate 필드에 대해 어떻게해야합니까 ??

나는 노력했다

@JsonDeserialize(using = LocalDateDeserializer.class)
@JsonSerialize(using = LocalDateSerializer.class)
private LocalDate dateOfBirth;  

작동하지 않았다.

누군가 올바른 방법이 무엇인지 알려주세요 ..

아래는 종속성입니다

<dependency>
    <groupId>org.jboss.resteasy</groupId>
    <artifactId>jaxrs-api</artifactId>
     <version>3.0.9.Final</version>
</dependency>
<dependency>
    <groupId>com.fasterxml.jackson.jaxrs</groupId>
    <artifactId>jackson-jaxrs-json-provider</artifactId>
    <version>2.4.2</version>
</dependency>
<dependency>
    <groupId>com.wordnik</groupId>
    <artifactId>swagger-annotations</artifactId>
    <version>1.3.10</version>
</dependency>



답변

주석을 사용 하여이 작업을 간단하게 수행 할 수 없었습니다. 작동 시키려면 ContextResolverfor을 만든 ObjectMapper다음 write-date-as-time-stamp를 false로 설정 해야하는 한 가지 경고와 함께 JSR310Module( 업데이트 : 이제 JavaTimeModule대신 )를 추가했습니다. JSR310 모듈 설명서를 참조하십시오 . 여기 내가 사용한 것의 예가 있습니다.

의존

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

참고 :이 문제에 직면 한 한 가지 문제는 jackson-annotation버전 2.3.2를 사용하여 다른 종속성으로 가져온 버전 이 필요하다는 것 jsr310입니다. 일어난 일은 NoClassDefFound for ObjectIdResolver인데 2.4 클래스입니다. 포함 된 종속성 버전을 정렬해야했습니다.

ContextResolver

import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.SerializationFeature;
import com.fasterxml.jackson.datatype.jsr310.JSR310Module;
import javax.ws.rs.ext.ContextResolver;
import javax.ws.rs.ext.Provider;

@Provider
public class ObjectMapperContextResolver implements ContextResolver<ObjectMapper> {
    private final ObjectMapper MAPPER;

    public ObjectMapperContextResolver() {
        MAPPER = new ObjectMapper();
        // Now you should use JavaTimeModule instead
        MAPPER.registerModule(new JSR310Module());
        MAPPER.configure(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS, false);
    }

    @Override
    public ObjectMapper getContext(Class<?> type) {
        return MAPPER;
    }
}

리소스 클래스

@Path("person")
public class LocalDateResource {

    @GET
    @Produces(MediaType.APPLICATION_JSON)
    public Response getPerson() {
        Person person = new Person();
        person.birthDate = LocalDate.now();
        return Response.ok(person).build();
    }

    @POST
    @Consumes(MediaType.APPLICATION_JSON)
    public Response createPerson(Person person) {
        return Response.ok(
                DateTimeFormatter.ISO_DATE.format(person.birthDate)).build();
    }

    public static class Person {
        public LocalDate birthDate;
    }
}

테스트

curl -v http://localhost:8080/api/person
결과: {"birthDate":"2015-03-01"}

curl -v -POST -H "Content-Type:application/json" -d "{\"birthDate\":\"2015-03-01\"}" http://localhost:8080/api/person
결과: 2015-03-01


JAXB 솔루션 은 여기 를 참조하십시오 .

최신 정보

JSR310Module잭슨의 버전 2.7의 추천되지 않습니다. 대신 모듈을 등록해야합니다 JavaTimeModule. 여전히 동일한 종속성입니다.


답변

@JsonSerialize와 @JsonDeserialize는 나를 위해 잘 작동했습니다. 추가 jsr310 모듈을 가져올 필요가 없습니다.

@JsonDeserialize(using = LocalDateDeserializer.class)
@JsonSerialize(using = LocalDateSerializer.class)
private LocalDate dateOfBirth;

디시리얼라이저 :

public class LocalDateDeserializer extends StdDeserializer<LocalDate> {

    private static final long serialVersionUID = 1L;

    protected LocalDateDeserializer() {
        super(LocalDate.class);
    }


    @Override
    public LocalDate deserialize(JsonParser jp, DeserializationContext ctxt)
            throws IOException, JsonProcessingException {
        return LocalDate.parse(jp.readValueAs(String.class));
    }

}

시리얼 라이저 :

public class LocalDateSerializer extends StdSerializer<LocalDate> {

    private static final long serialVersionUID = 1L;

    public LocalDateSerializer(){
        super(LocalDate.class);
    }

    @Override
    public void serialize(LocalDate value, JsonGenerator gen, SerializerProvider sp) throws IOException, JsonProcessingException {
        gen.writeString(value.format(DateTimeFormatter.ISO_LOCAL_DATE));
    }
}


답변

ObjectMapper mapper = new ObjectMapper();
mapper.registerModule(new JavaTimeModule());
mapper.configure(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS, false);

나를 위해 잘 작동합니다.


답변

Spring Boot 웹 앱에서 Jackson JSR 310 버전 “2.8.5”

compile "com.fasterxml.jackson.core:jackson-databind:2.8.5"
runtime "com.fasterxml.jackson.datatype:jackson-datatype-jsr310:2.8.5"

@JsonFormat작품 :

import com.fasterxml.jackson.annotation.JsonFormat;

@JsonFormat(shape = JsonFormat.Shape.STRING, pattern = "yyyy-MM-dd")
private LocalDate birthDate;


답변

역 직렬화 및 직렬화를 지원하는 가장 간단한 솔루션은 다음과 같습니다.

import com.fasterxml.jackson.annotation.JsonFormat;
import com.fasterxml.jackson.databind.annotation.JsonDeserialize;
import com.fasterxml.jackson.databind.annotation.JsonSerialize;
import com.fasterxml.jackson.datatype.jsr310.deser.LocalDateDeserializer;
import com.fasterxml.jackson.datatype.jsr310.ser.LocalDateSerializer;

@JsonFormat(shape = JsonFormat.Shape.STRING, pattern = "dd/MM/yyyy")
@JsonDeserialize(using = LocalDateDeserializer.class)
@JsonSerialize(using = LocalDateSerializer.class)
private LocalDate dateOfBirth;

프로젝트에서 다음과 같은 종속성을 사용하는 동안

메이븐

<dependency>
   <groupId>com.fasterxml.jackson.core</groupId>
   <artifactId>jackson-databind</artifactId>
   <version>2.9.7</version>
</dependency>
<dependency>
   <groupId>com.fasterxml.jackson.datatype</groupId>
   <artifactId>jackson-datatype-jsr310</artifactId>
   <version>2.9.7</version>
</dependency>

그래들

compile "com.fasterxml.jackson.core:jackson-databind:2.9.7"
compile "com.fasterxml.jackson.datatype:jackson-datatype-jsr310:2.9.7"

ContextResolver, Serializer 또는 Deserializer의 추가 구현이 필요하지 않습니다.


답변

이후 LocalDateSerializer오히려 “년 – 월 – 일”기본적으로 (JSON 문자열)보다 나는 특별한 요구하지 않으려는 이후에 회전을 “[년, 월, 일]”(JSON 배열) ObjectMapper당신이 할 수있는 (설정을 LocalDateSerializer비활성화하면 문자열 을 생성 SerializationFeature.WRITE_DATES_AS_TIMESTAMPS하지만 추가 설정이 필요 하면 ObjectMapper), 다음을 사용합니다.

수입품 :

import com.fasterxml.jackson.databind.ser.std.ToStringSerializer;
import com.fasterxml.jackson.datatype.jsr310.deser.LocalDateDeserializer;

암호:

// generates "yyyy-MM-dd" output
@JsonSerialize(using = ToStringSerializer.class)
// handles "yyyy-MM-dd" input just fine (note: "yyyy-M-d" format will not work)
@JsonDeserialize(using = LocalDateDeserializer.class)
private LocalDate localDate;

이제 new ObjectMapper()특별한 설정없이 객체를 읽고 쓰는 데 사용할 수 있습니다 .


답변

크리스토퍼의 업데이트에 대한 답변입니다.

버전 2.6.0 부터

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

사용 JavaTimeModule 대신 JSR310Module을 (사용되지 않음).

@Provider
public class ObjectMapperContextResolver implements ContextResolver<ObjectMapper> {
    private final ObjectMapper MAPPER;

    public ObjectMapperContextResolver() {
        MAPPER = new ObjectMapper();
        MAPPER.registerModule(new JavaTimeModule());
        MAPPER.configure(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS, false);
    }

    @Override
    public ObjectMapper getContext(Class<?> type) {
        return MAPPER;
    }
}

설명서 에 따르면 , 새로운 JavaTimeModule은 동일한 표준 설정을 사용하여 표준 시간대 ID를 사용하지 않고 직렬화를 기본값으로 설정하고 대신 ISO-8601 호환 표준 시간대 오프셋 만 사용합니다.

SerializationFeature를 사용하여 동작을 변경할 수 있습니다 .WRITE_DATES_WITH_ZONE_ID