@Transactional이고, Rollback 설정을 변경하지 않았음에도 Insert문이 실행되는 이유
427
작성한 질문수 2
학습하는 분들께 도움이 되고, 더 좋은 답변을 드릴 수 있도록 질문전에 다음을 꼭 확인해주세요.
1. 강의 내용과 관련된 질문을 남겨주세요.
2. 인프런의 질문 게시판과 자주 하는 질문(링크)을 먼저 확인해주세요.
(자주 하는 질문 링크: https://bit.ly/3fX6ygx)
3. 질문 잘하기 메뉴얼(링크)을 먼저 읽어주세요.
(질문 잘하기 메뉴얼 링크: https://bit.ly/2UfeqCG)
질문 시에는 위 내용은 삭제하고 다음 내용을 남겨주세요.
=========================================
[질문 템플릿]
1. 강의 내용과 관련된 질문인가요? (예/아니오)
2. 인프런의 질문 게시판과 자주 하는 질문에 없는 내용인가요? (예/아니오)
3. 질문 잘하기 메뉴얼을 읽어보셨나요? (예/아니오)
[질문 내용]
여기에 질문 내용을 남겨주세요.
안녕하세요 다름이 아니라 테스트 코드를 작성하여 결과를 확인하고 있었는데 교수님의 설명과 달리 Insert문이 계속해서 출력되고 있는 것이 의문이라 여쭤봅니다.
저는 rollback을 false로 하거나 em.flush를 따로 호출해주지 않았음에도 Insert문이 출력되는 이유가 궁금합니다.
참고로 저는 H2가 아니라 mysql 로컬 디비를 연결하여 개발 중인데 이 것이 사유가 되는걸까요?
spring:
datasource:
driver-class-name: com.mysql.cj.jdbc.Driver
url: jdbc:mysql://localhost:3306/testdb
username: root
password: ///
jpa:
hibernate:
ddl-auto: update
properties:
hibernate:
dialect: org.hibernate.dialect.MySQL8Dialect
format_sql: true
logging:
level:
org.hibernate.SQL: DEBUG
org.hibernate.type.descriptor.sql.BasicBinder: TRACE

package jpabook.jpashop.service;
import jpabook.jpashop.domain.Member;
import jpabook.jpashop.repository.MemberRepository;
import org.junit.Assert;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner;
import org.springframework.transaction.annotation.Transactional;
import static org.junit.Assert.*;
@RunWith(SpringRunner.class)
@SpringBootTest
@Transactional
public class MemberServiceTest {
//테스트니까 이런식으로 해도 됨.
@Autowired MemberService memberService;
@Autowired MemberRepository memberRepository;
@Test
public void 회원가입() throws Exception {
//~~걸 실행하면 ~~한 결과가 나와야 함, 검증해라 (given when then)
//given
Member member = new Member();
member.setName("kim");
//when
Long savedId = memberService.join(member);
//then
assertEquals(member, memberRepository.findOne(savedId));
}
@Test(expected = IllegalStateException.class)
public void 중복_회원_예외() throws Exception {
//given
Member member1 = new Member();
member1.setName("kim1");
Member member2 = new Member();
member2.setName("kim1");
//when
memberService.join(member1);
memberService.join(member2);
// try{
// memberService.join(member2); //예외가 발생
// }catch (IllegalStateException e){
// return;
// }
//then
fail("예외가 발생하야 한다");
}
}
답변 2
1
안녕하세요. sky980221님
JPA에서 엔티티를 보관하려면 먼저 ID(PK) 값을 구해야 합니다.
JPA의 ID 생성 전략 중에 IDENTITY 전략을 사용하게 되면 데이터베이스에 데이터를 저정해야 ID(PK)를 구할 수 있습니다. 따라서 먼저 insert 문이 실행될 수 있습니다.
아마도 데이터베이스가 달라서 전략이 달라진 것 같아요.
이 부분은 지금은 크게 고민하지 마시고, 이후에 JPA 기본편에 있는 기본 키 매핑을 학습해보시면 이해가 되실거에요.
감사합니다.
0
안녕하세요, 인프런 AI 인턴입니다.
sky980221님, 테스트 코드 실행 시 @Transactional과 rollback 설정을 변경하지 않았음에도 INSERT 문이 실행되는 이유에 대해 궁금하신 것 같습니다. 이 문제는 주로 테스트 환경에서 트랜잭션 관리 및 데이터베이스 설정과 관련이 있을 수 있습니다.
설명과 해결 방법:
@Transactional의 기본 동작:@Transactional이 Spring 테스트에서 사용되면 기본적으로 테스트 메소드가 끝난 후 롤백됩니다. 따라서 데이터가 실제로 DB에 반영되지 않지만, 트랜잭션 내에서는INSERT쿼리 실행이 가능합니다.
Hibernate의 플러시(flush) 매커니즘:
- Hibernate는 트랜잭션이 커밋되기 전에, 영속성 컨텍스트가 데이터베이스와 동기화되도록
flush를 수행합니다.flush는 실제로 SQL 쿼리를 실행하지만, 트랜잭션 내에서 롤백 설정이 되어 있다면 결국 데이터베이스에는 반영되지 않습니다.
- Hibernate는 트랜잭션이 커밋되기 전에, 영속성 컨텍스트가 데이터베이스와 동기화되도록
MySQL과 H2 설정 차이:
- H2와 같은 메모리 기반 데이터베이스는 테스트가 끝난 후 데이터가 사라지므로, MySQL과 비교했을 때
INSERT쿼리의 실행 여부가 눈에 띄지 않을 수 있습니다. 그러나 이는@Transactional의 동작과는 무관합니다.
- H2와 같은 메모리 기반 데이터베이스는 테스트가 끝난 후 데이터가 사라지므로, MySQL과 비교했을 때
로그 설정:
org.hibernate.SQL: DEBUG설정을 통해 모든 SQL 쿼리가 로깅됩니다. 이런 로그는 쿼리가 실제로 실행되었음을 의미하지는 않지만, Hibernate가 쿼리를 생성하여 데이터베이스에 전달했음을 보여줍니다.
참고하실 수 있는 기존 질문 & 답변 링크는 다음과 같습니다:
- 질문 제목: @Transactional 사용 시 롤백이 진행되지 않습니다. 여기서는 @Transactional 설정과 관련된 문제를 해결할 수 있는 실마리를 제공하고 있습니다.
이외에도, 트랜잭션 관리와 관련된 Spring 설정을 다시 한 번 확인해 보시고, 트랜잭션 경계가 명확히 설정되었는지 점검해 보시기 바랍니다.
저는 질문자님의 이해를 돕고자 지식공유자님의 기존 답변을 참고하여 유사한 답변 링크를 추천해드리고 있습니다. 현재 베타 기능으로 답변이 만족스럽지 않을 수 있는 점 양해 부탁드립니다. 🙏 추가적으로 궁금한 점이 있으시면, 이어서 질문해 주세요. 곧 지식공유자께서 답변해 주실 것입니다.
OrderServiceTest 상문주문 테스트 시 update 쿼리 문의
0
27
1
sdk 설정 오류
0
67
2
오탈자 - @Transactional
0
62
1
src/test/resources 테스트 경로 문제
0
56
1
상품 등록후 H2 db 출력 순서 바꿀 수 있나요?
0
70
1
MemberRepositoryTest 실행오류
0
90
1
boot 4.x >>> trasasction rolled back log & p6spy(영한님, 수업 자료 업데이트 해주시면 감사하겠습니다!!)
1
195
2
강의 마지막 QueryDSL 사용 부분 질문있습니다
1
153
2
클라이언트에서 isbn과 author 수정 요청을 한 경우에 대해 질문드립니다.
0
56
1
도메인 모델 패턴 vs 트랜잭션 스크립트 패턴
0
80
1
기본 생성자
0
67
1
h2 DB 연결시 jdbc url 변경 이유가 궁금합니다.
0
107
1
멤버서비스테스트 부분에서 막힙니다.
0
174
4
실무에서도 EntityManager를 이용해서 많이 작업하는 편일까요?
0
124
1
초반에 h2 다운로드 과정 꼭 필요한가요?
0
129
2
자신 필드에도 get으로 접근하는 이유가 있을까요?
0
121
1
24분 27초 연관관계 편의 메서드 위치
0
118
1
단건 주문만 가능하게 한건 의도한 부분이신가요?
0
114
2
빌드 툴, Gradle
0
63
1
h2연결은 된 것 같은데 엔티티 테이블까지 작성 후 확인해보아도 테이블이 안보입니다
0
81
2
Repository에서 EntityManager 주입 방식 차이
0
97
1
롬복과 사용자 정의 setter 메서드
0
77
1
주문 목록 조회 fetch join 질문드립니다
0
91
1
dirty checking 질문드립니다.
0
86
1





