[JPA] @Enumerated 와 AttributeConverter
@Enumerated
자바 enum 타입을 엔티티 클래스의 속성으로 사용할 수 있다.
@Enumerated
애노테이션에는 두 가지 EnumType이 존재한다.
EnumType.ORDINAL
: enum 순서 값을 DB에 저장EnumType.STRING
: enum 이름을 DB에 저장
예를 들어 enum이 다음과 같이 생겼다고 하자.
public enum Menu {
BULGOGI_PIZZA(10000, "PIZZA"),
CHEESE_PIZZA(8000, "PIZZA"),
POTATO_PIZZA(11000, "PIZZA"),
;
}
1) ORDINARY
@Entity
public class Order {
@Enumerated(EnumType.ORDINAL)
private Menu menu;
}
EnumType을 ORDINAL
로 지정하고 menu 속성에 Menu.BULGOGI_PIZZA
값을 셋팅하면 DB에 저장되는 값은 1이다.
그럼 Menu.CHEESE_PIZZA
값으로 셋팅하면 어떻게 될까?
Menu에 선언되어 있는 “순서”가 값이 되기 때문에 2가 DB에 저장된다.
2) STRING
@Entity
public class Order {
@Enumerated(EnumType.STRING)
private Menu menu;
}
EnumType을 STRING
으로 지정하면 BULGOGI_PIZZA
, CHEESE_PIZZA
“문자열” 자체가 저장된다.
(단점은 STRING
설정은 문자열 자체가 저장되기 때문에 DB 공간 낭비가 발생한다는 것이다…)
주의 사항
enum 값(name)으로 DB에 저장하면 더 알아보기 쉽다.
하지만 enum 값을 저장하는 과정에서 int가 string을 DB에 저장하게 되고 이는 메모리 용량을 좀 더 증가 시킬 수 있다.
하지만 이것은 데이터가 아주 많을 때의 문제이고 그렇지 않을 경우에는 큰 문제가 되지 않는다.
제일 큰 문제는 만일 enum에서는 BULGOGI_PIZZA
의 명칭을 BULGOGI_PZA
로 변경했을 경우에 맞는 name 값이 없어서 DB에서 해당 부분 데이터를 읽어오는데 에러가 발생한다는 것이다.
기본 값인 EnumType.ORDINAL
로 사용했을 때는 enum의 값들 순서가 바뀔 경우 데이터를 잘못 읽어오는 문제가 생긴다.
enum의 기존 값들의 순서가 바뀌거나 기존 name이 달라질 일이 없으면 신경쓰지 않고 @Enumerated
어노테이션을 사용해도 되지만(새로운 값이 추가되는 경우는 상관 없다.)
그렇지 않을 경우에는 별도 Converter를 생성해 enum의 값을 DB에 넣을 때와 enum의 값을 DB에서 읽어올 때에 작동하는 함수를 오버라이딩해서 바뀐 데이터에 대해 재처리 하는 로직을 작성해야 한다.
@Converter
위에서 @Enumerated
어노테이션을 사용한다면 아래 두 가지 상황에서 문제가 발생할 수 있다.
- 데이터 삭제 등으로 order “순서”가 변경 되었을 경우
- enum의 “값”이 변경 되었을 경우
따라서 enum를 다룰 때 @Enumerated
어노테이션 대신,
AttributeConverter을 구현한 class에 @Converter
어노테이션을 사용하면 DB에 원하는 enum의 값을 저장할 수 있다.
@Converter이란?
@Converter
어노테이션은 엔티티의 데이터를 DB에서 가져올 때 변환하는 방법에 대한 로직과 DB에서 엔티티의 데이터를 가져올 때의 변환하는 방법에 대한 로직을 정의해놓은 변환하는 클래스에 붙이는 어노테이션이다.
@Converter(autoApply = true)
로 제일 많이 쓰이는데
이 뜻은 해당 변환 클래스에 지정된 타입에 대해서는 모두 해당 변환 클래스의 메소드를 이용해 DB와의 통신에서 값을 변환하겠다는 것이고,
autoApply
옵션은 기본 값이 false
로 따로 XML과 같은 설정 파일에 해당 변환 클래스를 사용할 곳에서만 변환이 이루어지게 된다.
AttributeConverter 인터페이스 구현
AttributeConverter를 implements한 클래스를 생성하면 아래의 두 함수를 오버라이딩하게 되어 있다.
@Converter(autoApply = true)
public class MenuConverter implements AttributeConverter<Menu, String>{
@Override
public String convertToDatabaseColumn(Menu menu) {
...
}
@Override
public Menu convertToEntityAttribute(String menuName) {
...
}
}
convertToDatabaseColumn(Menu menu)
: enum을 DB에 어떤 값으로 넣을 것인지 정의converToEntityAttribute(String menuName)
: DB에서 읽힌 값에 따라 어떻게 enum랑 매칭 시킬 것인지 정의
AttributeConverter를 구현할 때 AttributeConverter안에 어떤 엔티티를 사용할 것이며,
해당 엔티티가 DB에 어떤 타입으로 저장되어 있는지 넣도록 되어 있다.
AttributeConverter<Menu, String>
에서 첫 번째에는 변환에 사용할 엔티티, 여기서는 enum class를 넣으면 되고
여기 예제에서는 메뉴의 이름을 텍스트로 DB에 저장하므로 String이라고 타입을 지정해줬다.
예를 들면 아래와 같이 두 함수를 오버라이딩 할 수 있다.
@Converter(autoApply = true)
public class MenuConverter implements AttributeConverter<Menu, String>{
@Override
public String convertToDatabaseColumn(Menu menu) {
if(menu == null) {
return null;
}
return menu.getMenuName();
}
@Override
public Menu convertToEntityAttribute(String menuName) {
if(menuName == null) {
return null;
}
return Stream.of(Menu.values())
.filter(m -> m.getMenuName().equals(menuName))
.findFirst()
.orElse(null);
}
}
우선 convertToDatabaseColumn
(Enum => DB)에서는 menu가 null일 경우네는 null을,
null이 아닐 경우 enum에서 메뉴이름을 정의해둔 getMenuName() 함수로 return 하도록 정의했다.
convertToEntityAttribute
(DB => Enum)에서는 마찬가지로 menuName이 null이 아닐 경우에는 Stream.of()
함수를 통해 enum에 있는 값들을 모두 가져오고 거기서 filter
함수를 사용해서 getMenuName()
으로 나오는 메뉴 이름과 일치하는 것을 찾았다.
그리고 찾은 것에서 findFirst()
를 해서 찾은 것의 첫 번째를 return하고 orElse()
를 통해 많이 일치되는게 없을 경우 null 값을 return 하도록 했다.
위 코드에서는 별도로 enum의 값이 변환 된 것에 대한 커스텀 로직을 넣지 않았지만
enum의 값이 변환되어 미치못하게 예외 로직을 짜서 넣어야할 경우
위의 convertToDatabaseColumn()
, convertToEntityAttribute()
함수에 해당 로직을 적절하게 짜 넣으면
enum과 DB 사이에서 값이 적절하게 변환되어 에러 없이 사용할 수 있게 된다.
AttributeConverter로 구현 Converter를 구현하면 Enum 클래스의 @JsonCreator를 대신할 수 있는가?
결론부터 말하면 아니다.
@JsonCreator
는 entity를 Controller에서 RequestBody와 같이 JSON 형태로 받을 때 동작이 일어난다.
하지만 AttributeConverter는 entity와 DB 사이의 변환 역할을 해주는 함수를 구현하는 인터페이스이다.
따라서 request에서 entity의 값이 parse 되는데에는 Converter가 아닌 @JsonCreator
와 같은 방법을 사용해야 한다.
Ref.
- https://lng1982.tistory.com/280
- https://lng1982.tistory.com/279 - AttributeConverter
- https://pamyferret.tistory.com/6
- https://pamyferret.tistory.com/7 - AttributeConverter (이거 좋음)
- https://minji6119.tistory.com/42 - AttributeConverter
- https://techblog.woowahan.com/2600/ - 우형 기술블로그
- https://devvkkid.tistory.com/270 - AttributeConverter (이거 보기)
💛 개인 공부 기록용 블로그입니다. 👻