Search
Duplicate

스프링 데이터 JPA 분석

스프링 데이터 JPA 구현체 분석

스프링 데이터 JPA가 제공하는 공통 인터페이스의 구현체
org.springframework.data.jpa.repository.support.SimpleJpaRepository

SimpleJpaRepository

Code

save() 메서드

새로운 엔티티면 저장(persist)
새로운 엔티티가 아니면 병합(merge)

→ 그럼 어떻게 새로운 엔티티인지 확인 가능한가?

새로운 엔티티를 구별하는 방법

→ 새로운 엔티티인지 판단하는 기본 전략은 아래와 같다.

식별자가 객체일때(Integer, Long...) null 로 판단한다.
식별자가 자바 기본타입(Primitive type : int, long...)일 때 0 으로 판단한다.
Persistable 인터페이스를 구현해서 판단 로직 변경도 가능하다.
package org.springframework.data.domain; public interface Persistable<ID> { ID getId(); boolean isNew(); }
Java
복사
참고: JPA 식별자 생성전략이 @GenerateValue 일 경우 save()호출 시점에는 식별자가 없기 때문에 새로운 엔티티로 인식해서 정상동작한다. 하지만, JPA 식별자 생성 전략이 @Id 만 사용해서 직접 할당하는 경우 식별자값이 있는 상태로 save()를 호출하게 된다. 이 경우 save()에서는 새로운 엔티티가 아니라고 판단해서 병합(merge)가 호출된다. 그런데 병합(merge)은 우선 DB를 호출해서 값을 확인 후, DB에 값이 없으면 새로운 엔티티로 호출 할 뿐아니라 엔티티의 모든 값을 교체해버리기 때문에 사이드이펙트(의도치 않은 값 변경)가 발생할 수 있다. 따라서 Persistable를 사용해서 새로운 엔티티 확인 여부를 직접 구현하는게 효과적이다. 해당 강의에서는 @CreatedDate을 조합해서 사용하면 새로운 엔티티 여부를 판단하는게 쉬워진다. (@CreatedDate에 값이 없으면 새로운 엔티티로 판단)
@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; } }
Java
복사