실전! 스프링 데이터 JPA 수업을 듣고 정리한 내용입니다.
✔️ 스프링 데이터 JPA가 제공하는 공통 인터페이스의 구현체
SimpleJpaRepository
org.springframework.data.jpa.repository.support.SimpleJpaRepository
@Repository
적용 : JPA 예외를 스프링이 추상화한 예외로 변환
@Transactional
트랜잭션 적용
SimpleJpaRepository
에 구현된 메서드들이 이미 트랜잭션 안에서 처리되도록 구성되었기 때문에 트랜잭션이 없어도 데이터를 등록, 변경이 가능했다. (트랜잭션이 리포지토리 계층에 걸려있는 것이다.)@Transactional(readOnly = true)
readOnly = true
옵션을 사용하면 약간의 성능 향상을 얻을 수 있다. flush()
를 할 필요가 없다.save()
는 매우 중요하다!
persist
)merge
)
🔔
save() 메서드
- 새로운 엔티티면 저장(
persist
)- 새로운 엔티티가 아니면 병합(
merge
)
🔔 새로운 엔티티를 판단하는 기본 전략
- 식별자가 객체일 때
null
로 판단- 식별자가 자바 기본 타입일 때
0
으로 판단Persistable
인터페이스를 구현해서 판단 로직 변경 가능
Persistable
💡 참고
- JPA 식별자 생성 전략이
@GenerateValue
면save()
호출 시점에 식별자가 없으므로 새로운 엔티티로 인식해서 정상 동작한다.- 그런데 JPA 식별자 생성 전략이
@Id
만 사용해서 직접 할당이면 이미 식별자 값이 있는 상태로save()
를 호출한다. 따라서 이 경우merge()
가 호출된다.merge()
는 우선 DB를 호출해서 값을 확인하고, DB에 값이 없으면 새로운 엔티티로 인지하므로 매우 비효율적이다.- 따라서
Persistable
를 사용해서 새로운 엔티티 확인 여부를 직접 구현하게는 효과적이다.
💡 참고
등록시간(@CreatedDate
)을 조합해서 사용하면 이 필드로 새로운 엔티티 여부를 편리하게 확인할 수 있다. (@CreatedDate
에 값이 없으면 새로운 엔티티로 판단한다.)
✔️ Persistable 구현
package study.datajpa.entity;
import lombok.AccessLevel;
import lombok.NoArgsConstructor;
import org.springframework.data.annotation.CreatedDate;
import org.springframework.data.domain.Persistable;
import org.springframework.data.jpa.domain.support.AuditingEntityListener;
import javax.persistence.Entity;
import javax.persistence.EntityListeners;
import javax.persistence.Id;
import java.time.LocalDateTime;
@Entity
@EntityListeners(AuditingEntityListener.class)
@NoArgsConstructor(access = AccessLevel.PROTECTED)
public class Item implements Persistable<String> {
@Id
private String id;
@CreatedDate
private LocalDateTime createDate;
public Item(String id) {
this.id = id;
}
@Override
public String getId() {
return id;
}
@Override
public boolean isNew() {
return createDate == null;
}
}
✏️ debug 모드 실행하기
- 빨간색 아이콘 만들고(소스 줄 클릭시 빨간 색 아이콘 만들어진다.)
ctrl
+d
단축키 (디버그 모드 실행)