[Spring Data JPA] - 벌크성 수정 쿼리

홍정완·2022년 7월 15일
0

JPA

목록 보기
19/38
post-thumbnail
post-custom-banner

벌크성 쿼리란?


  • 벌크성 쿼리란 DB에서 여러 개의 레코드를 한 번에 추가/수정/삭제하는 쿼리



// 순수 JPA를 사용한 벌크성 수정 쿼리

public int bulkAgePlus(int age) {

 	int resultCount = em.createQuery(
		 "update Member m set m.age = m.age + 1 where m.age >= :age")
		 .setParameter("age", age)
		 .executeUpdate();
         
	 return resultCount;
}

// 순수 JPA를 사용한 벌크성 수정 쿼리 테스트

@Test
public void bulkUpdate() throws Exception {
 
 	//given
 	memberJpaRepository.save(new Member("member1", 10));
 	memberJpaRepository.save(new Member("member2", 19));
	memberJpaRepository.save(new Member("member3", 20));
	memberJpaRepository.save(new Member("member4", 21));
 	memberJpaRepository.save(new Member("member5", 40));
 
 
 	//when
	int resultCount = memberJpaRepository.bulkAgePlus(20);
 
 
 	//then
	assertThat(resultCount).isEqualTo(3);
}



// Spring Data JPA를 사용한 벌크성 수정 쿼리

@Modifying(clearAutomatically = true)
@Query("update Member m set m.age = m.age + 1 where m.age >= :age")
int bulkAgePlus(@Param("age") int age);

// Spring Data JPA를 사용한 벌크성 수정 쿼리 테스트

@Test
public void bulkUpdate() throws Exception {

	//given
	memberRepository.save(new Member("member1", 10));
	memberRepository.save(new Member("member2", 19));
	memberRepository.save(new Member("member3", 20));
 	memberRepository.save(new Member("member4", 21));
	memberRepository.save(new Member("member5", 40));
 
 
 	//when
 	int resultCount = memberRepository.bulkAgePlus(20);
 
 
 	//then
	assertThat(resultCount).isEqualTo(3);
}



  • 벌크성 수정, 삭제 쿼리는 @Modifying 어노테이션 사용
    • 사용하지 않으면 다음 예외 발생
    • org.hibernate.hql.internal.QueryExecutionRequestException: Not supported for DML operations

  • 벌크성 쿼리를 실행하고 나서 영속성 컨텍스트 초기화 : @Modifying(clearAutomatically = true)

    • 옵션의 기본값은 false

    • 옵션 없이 회원을 findById로 다시 조회하면 영속성 컨텍스트에 과거 값이 남아서 문제가 될 수 있다. 만약 다시 조회해야 하면 꼭 영속성 컨텍스트 초기화



  • 벌크 연산은 영속성 컨텍스트를 무시하고 실행하기 때문에, 영속성 컨텍스트에 있는 엔티티의 상태와
    DB에 엔티티 상태가 달라질 수 있다.



✅ 권장하는 방안

  • 영속성 컨텍스트에 엔티티가 없는 상태에서 벌크 연산을 먼저 실행
  • 부득이하게 영속성 컨텍스트에 엔티티가 있으면, 벌크 연산 직후 영속성 컨텍스트를 초기화
profile
습관이 전부다.
post-custom-banner

0개의 댓글