Java/Spring

[Spring] 변경 감지(Dirty checking)와 병합(merge)

SeungbeomKim 2023. 3. 8. 11:26
반응형

 

JPA에서 변경 감지와 병합 두 가지 요소를 알기 전에 준영속 엔티티를 알고 있어야 합니다.

준영속 엔티티에 대해 설명드리겠습니다.

 

준영속 엔티티란 영속성 컨텍스트가 더 이상 관리하지 않은 엔티티입니다. 

@PostMapping("items/{itemId}/edit")
    public String updateItem(@PathVariable String itemId, @ModelAttribute("form") BookForm form) {
        Book book = new Book();
        book.setId(form.getId());
        book.setName(form.getName());
        book.setPrice(form.getPrice());
        book.setStockQuantity(form.getStockQuantity());
        book.setAuthor(form.getAuthor());
        book.setIsbn(form.getIsbn());

        itemService.saveItem(book);
        return "redirect:/items"
    }

여기에서 book 객체는 이미 DB에 한 번 저장이 되었기에 식별자가 존재합니다. 이렇게 엔티티가 JPA가 식별할 수 있는 식별자를 가지고 있으면 준영속 엔티티라고 볼 수 있습니다. 이러한 객체는 JPA가 관리하지 않으며 JPA에서 업데이트할 수 있는 근거가 없습니다.

그래서 이러한 "준영속 상태의 엔티티는 데이터를 어떻게 변경할 수 있을까?"라는 관점에서 변경 감지와 병합 개념이 도입되었습니다.

 

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

 

1. 변경 감지 기능 

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

@Transactional 어노테이션에 의해 트랜잭션이 커밋되는 시점에 JPA는 flush를 날립니다. flush는 DB에 변경된 사항이 있는지 check해주는 기능입니다. 위의 과정들에 의해서 준영속 엔티티를 수정할 수 있게 됩니다.

 

2. 병합 기능

@Transactional
void update(Item itemParam) { 
	//itemParam: 파리미터로 넘어온 준영속 상태의 엔티티 
	Item mergeItem = em.merge(itemParam);
}

 

병합은 준영속 상태의 엔티티를 영속 상태로 변경할 때 사용하는 기능입니다.

병합 동작 방식

  1. merge() 메서드 실행
  2. Parameter로 넘어온 준영속 엔티티의 식별자 값으로 1차 캐시에서 엔티티를 조회합니다. -> 1차 캐시에 엔티티가 없는 경우에는 DB에서 엔티티를 조회하고, 1차 캐시에 저장
  3. 조회한 영속 엔티티(mergeItem)에 item 엔티티의 값을 채워 넣는다. (item 엔티티의 모든 값을 mergeItem에 밀어 넣습니다) == 병합하는 과정
  4. 영속 상태인  mergeItem 반환

<주의사항> 

변경 감지 기능을 사용하면 원하는 속성만 변경할 수 있지만, 병합을 사용하면 모든 속성이 변경됩니다. 그래서 병합 시 값이 없으면 null로 업데이트 되기에 엄청 위험합니다. (병합은 모든 필드를 교체) 예시를 들어, Book 객체에 isbn필드에 값이 null인 경우 데이터베이스에 null로 업데이트가 됩니다. 이러한 이유 때문에, 변경 감지 기능을 사용하여 변경할 필드 부분들만 업데이트 해줘야 합니다. 

 

반응형