RequestDto는 왜 기본생성자가 없을 때 에러를 반환할까?
일반적으로 클라이언트가 Request Body를 통해서 서버에 API 요청할 때, 서버에서는 이걸 RequestDto
에 매핑해서 전달 받는다. (ex. public ResponseDto call(@RequestBody RequestDto requestDto) { ... }
)
여기서 만약 RequestDto에 기본 생성자를 구현하지 않았다면, 어떤 일이 벌어지게 될까?
직렬화/역직렬화는 jackson 라이브러리를 통해서 수행된다
클라이언트 - 서버 간 요청/응답에 대한 Body 데이터와 객체 바인딩은 jackson
라이브러리를 통해 수행한다.
(jackson
라이브러리는 spring-boot-starter-web
라이브러리에 포함되어 있다)
클라이언트 -> 서버의 경우는 역직렬화, 서버 -> 클라이언트의 경우는 직렬화가 이루어진다.
이때, Jackson2HttpMessageConverter
가 내부적으로 ObjectMapper
를 사용해서 데이터를 처리한다.
ObjectMapper
는 Getter(public)
/ Setter
을 통해 프로퍼티명을 확인한다.
(BeanDeserializerFactory
에서 프로퍼티명 찾는 작업 수행, BeanDeserializer
에서 DTO에 값을 주입)
값을 주입하는 것은 java.lang.reflect
패키지를 사용하기 때문에 setter
없이도 가능하다. (Field
자료형)
(Getter, Setter 중 하나만 존재하면 값 주입이 가능)
그래서 ObjectMapper에서 기본 생성자를 어떻게 쓰고, 왜 없을 때 안되는건데?
내부 코드는 크게 이렇게 동작한다고 한다.
-
기본 생성자가 있는 경우
-
super.createUsingDefault()
return _defaultCreator.call()
-
-
기본 생성자가 없는 경우
-
deserializeFromObjectUsingNonDefault()
-
deletegateSerializer
사용
-
-
여기서, 기본 생성자가 없는 경우 deletegateSerializer
를 사용해야 하는데, 이는 DTO에 생성을 위임하거나 프로퍼티를 별도로 설정이 필요하다.
이러한 조건이 만족되지 않는다면 위와 같은 예외를 반환하는 것이다.
DTO에 생성을 위임하거나 프로퍼티를 별도 설정하는 방법은 Json 관련 어노테이션을 사용하는 것이다.
(@JsonProperty
, @JsonAutoDetect
, @JsonCreator
)