스프링 데이터 JPA 분석
구현체
org.springframework.data.jpa.repository.support.SimpleJpaRepository
스프링 데이터 JPA가 제공하는 공통 인터페이스의 구현체
@Repository
@Transactional(readOnly = true)
public class SimpleJpaRepository<T, ID> implements JpaRepositoryImplementation<T, ID> {
}
@Repository
JPA 예외를 스프링이 추상화 한 예외로 변환한다.
기술이 바뀌어도 영향을 주지 않도록 분리할 수 있다.
@Transactional
@Repository
@Transactional(readOnly = true)
public class SimpleJpaRepository<T, ID> implements JpaRepositoryImplementation<T, ID> {
// 변경 사항은 읽기 전용이 아닌 트랜잭션을 사용하도록 내부적으로 설정되어 있다.
@Override
@Transactional
public void deleteAll(Iterable<? extends T> entities) {
Assert.notNull(entities, "Entities must not be null!");
for (T entity : entities) {
delete(entity);
}
}
}
트랜잭션을 적용한다.
JPA의 모든 변경은 트랜잭션 안에서 동작한다.
스프링 데이터 JPA는 등록, 수정, 삭제 등 변경 메서드를 트랜잭션 처리한다.
서비스 계층에서 트랜잭션을 시작하지 않으면
리파지토리에서 트랜잭션을 시작한다.
그래서 예제에서 @Transactional 애너테이션이 없어도 데이터 등록, 변경이 가능했다.
트랜잭션이 리포지토리 계층에 이미 걸려있는 것이다.
서비스 계층에서 트랜잭션을 시작하면
리파지토리는 해당 트랜잭션을 전파 받아서 사용한다.
@Transactional(readOnly = true)
단순 조회만 하고 변경은 하지 않는 트랜잭션에 사용한다.
flush를 생략해서 약간의 성능 향상을 얻을 수 있다.
JPA 책 15.4.2를 참고한다.
새로운 엔티티를 구별하는 방법
save()
@Repository
@Transactional(readOnly = true)
public class SimpleJpaRepository<T, ID> implements JpaRepositoryImplementation<T, ID> {
@Transactional
@Override
public <S extends T> S save(S entity) {
Assert.notNull(entity, "Entity must not be null.");
if (entityInformation.isNew(entity)) {
em.persist(entity);
return entity;
} else {
return em.merge(entity);
}
}
}
새로운 엔티티는 저장한다.
객체의 식별자가 null 일 때
기본 타입의 식별자가 0일 때
Persistable 인터페이스를 구현해서 판단 로직을 변경할 수 있다.
새로운 엔티티가 아니면 병합한다.
기존 값을 새로 들어온 데이터로 교체해버린다.
DB에서 select를 무조건 한 번 한다는 단점이 있다.
따라서 데이터 변경은 병합보다는 변경 감지를 활용하는 게 좋다.
병합은 영속 상태 엔티티가 잠시 영속 상태를 벗어났다가 다시 영속 상태가 되어야할 때 사용한다.
@Entity
@Getter
public class Item {
@Id
@GeneratedValue
private Long id;
}

새로운 객체이므로 id가 null이다.
@Entity
@Getter
@NoArgsConstructor(access = AccessLevel.PROTECTED)
public class Item {
@Id
private String id;
public Item(String id) {
this.id = id;
}
}
@SpringBootTest
class ItemRepositoryTest {
@Autowired
ItemRepository itemRepository;
@Test
void save() {
Item item = new Item("A");
itemRepository.save(item);
}
}

select item0_.id as id1_0_0_
from item item0_
where item0_.id = 'A';
insert into item (id)
values ('A');
식별자에 값이 있으므로 병합한다.
DB에 값이 있다는 걸 가정하는 것이기 때문에 일단 있는지 쿼리한다.
Persistable
package org.springframework.data.domain;
public interface Persistable<ID> {
ID getId();
boolean isNew();
}
어떤 이슈로 임의의 ID를 직접 생성해야 한다면 Persistable를 implement 한다.
@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;
}
}
insert into item (created_date, id)
values ('2022-05-22T16:23:28.873+0900', 'A');
등록 시간인 @CreatedDate를 조합해서 사용하면 새 객체인지 쉽게 구별할 수 있다.
똑같은 테스트를 돌리면 select 없이 insert만 치는 것을 확인할 수 있다.
Last updated
Was this helpful?