ABOUT ME

-

Today
-
Yesterday
-
Total
-
  • [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;
        }
    }

    댓글

Designed by Tistory.