(JPA) 변경 감지와 병합(Merge)

업데이트:
1 분 소요

변경 감지와 병합(merge) (JPA)

1. 준영속 엔티티

  • 영속성 컨텍스트가 더는 관리하지 않은 엔티티를 말한다.
  • 여기서는 itemService.saveItem(book)에서 수정을 시도하는 Book객체다.
  • Book객체는 이미 DB에 한번 저장되어서 식별자가 존재한다.
  • 이렇게 임의로 만들어낸 엔티티도 기존 식별자를 가지고 있으면 준영속 엔티티로 볼 수 있다.

2. 준영속 엔티티를 수정하는 2가지 방법

1) 변경 감지 기능 사용

  • 영속성 컨텍스트에서 엔티티를 다시 조회한 후에 데이터를 수정하는 방법
  • 트랜잭션 안에서 엔티티를 다시 조회, 변경할 값 선택 ->
    • 트랜잭션 커밋 시점에 변경 감지(Dirty Checking)이 동작해서 데이터베이스에 UPDATE SQL 실행
    @Transactional
    void update(Item itemParam) { //itemParam: 파리미터로 넘어온 준영속 상태의 엔티티
    Item findItem = em.find(Item.class, itemParam.getId()); //같은 엔티티를 조회한 다.
    findItem.setPrice(itemParam.getPrice()); //데이터를 수정한다. }

2) 병합(merge)사용

  • 병합은 준영속 상태의 엔티티를 영속 상태로 변경할 때 사용하는 기능이다.
    @Transactional
    void update(Item itemParam) { //itemParam: 파리미터로 넘어온 준영속 상태의 엔티티
    Item mergeItem = em.merge(item);
    }

3. 병합 동작 방식

alt

  1. merge()를 실행한다.
  2. 파라미터로 넘어온 준영속 엔티티의 식별자 값으로 1차 캐시에서 엔티티를 조회한다.
    • 만약 1차 캐시에 엔티티가 없으면 데이터베이스에서 엔티티를 조회하고, 1차 캐시에 저장한다.
  3. 조회한 영속 엔티티(mergeMember)에 member엔티티의 값을 채워 넣는다.
    • member 엔티티의 모든 값을 mergeMember에 밀어 넣는다. 이때 mergeMember의 “회원1”이라는
      이름이 “회원명 변경”으로 바뀐다.
  4. 영속 상태인 mergeMember를 반환한다.

4. 병합시 동작 방식을 간단히 정리

  • 준영속 엔티티의 식별자 값으로 영속 엔티티를 조회한다.
  • 영속 엔티티의 값을 준영속 엔티티의 값으로 모두 교체한다. (병합한다.)
  • 트랜잭션 커밋 시점에 변경 감지 기능이 동작해서 데이터베이스에 UPDATE SQL 실행

주의

  • 변경 감지 기능을 사용하면 원하는 속성만 선택해서 변경할 수 있지만, 병합을 사용하면 모든 속성이 변경
  • 병합시 값이 없으면 null로 업데이트 할 위험도 있다. (병합은 모든 필드 교체)

출처 : 스프링부트 JPA 활용

태그:

카테고리:

업데이트:

댓글남기기