-
[Spring] 새로운 엔티티를 구별하는 방법Spring 2022. 1. 1. 00:00
spring data jpa의 save함수는 구조는 이런식이다.
@Transactional public <S extends T> S save(S entity) { if (entityInformation.isNew(entity)) { em.persist(entity); return entity; } else { return em.merge(entity); } }
- 새로운 엔티티면 저장(persist)
- 새로운 엔티티가 아니면 병합(merge)
그렇다면 새로운 엔티티를 구별하는 기준은 무엇일까?
- 식별자가 객체일 때 null로 판단
- 식별자가 자바 기본타입일 때 0으로 판단
- Persistable 인터페이스를 구현해서 판단로직 변경 가능
JPA 식별자 생성 전략이 @GenerateValue 면 save() 호출 시점에 식별자가 없으므로 새로운 엔티티로 인식해서 정상 동작한다.
그런데 JPA 식별자 생성 전략이 @Id 만 사용해서 직접 할당이면 이미 식별자 값이 있는 상태로 save() 를 호출한다.
따라서 이 경우 merge() 가 호출된다.
merge() 는 우선 DB를 호출해서 값을 확인하고, DB에 값이 없으면 새로운 엔티티로 인지하므로 매우 비효율적이다.
따라서 Persistable 를 사용해서 새로운 엔티티 확인 여부를 직접 구현하게는 효과적이다.
참고로 등록시간( @CreatedDate )을 조합해서 사용하면 이 필드로 새로운 엔티티 여부를 편리하게 확인할 수 있다.
(@CreatedDate에 값이 없으면 새로운 엔티티로 판단)
Persisatable 구조
public interface Persistable<ID> { ID getId(); boolean isNew(); }
Persisatable을 이용한 구현
@Entity @EntityListeners(AuditingEntityListener.class) @NoArgsConstructor(access = AccessLevel.PROTECTED) public class Item implements Persistable<String> { @Id private String id; @CreatedDate private LocalDateTime createdDate; public Item(String id) { this.id = id; } @Override public String getId() { return id; } @Override public boolean isNew() { return createdDate == null; } }
'Spring' 카테고리의 다른 글
[Spring] querydsl - 검색 조건 쿼리 (0) 2022.01.03 [Spring] JPAQueryFactory를 필드로 선언하면 어떻게 될까? (0) 2022.01.03 [Spring] 한 페이지에 페이징 정보가 둘 이상일 경우 (0) 2021.12.31 [Spring] 페이징과 정렬 (0) 2021.12.31 [Spring] Auditing (0) 2021.12.30