MemberServiceTest - 테스트에서 @Transaction의 롤백기능이 수행되지 않습니다
안녕하세요.
올려주신 강의 잘 듣던 중 궁금한 점이 생겨 질문드립니다.
@Transactional을 테스트 케이스에서 쓸 경우에는 커밋이 안되고 롤백을 한다고 하셨는데, MemberServiceTest에서 만든 2개의 테스트 중 중복_회원_예외() 메서드 실행 시에는 아래 로그에 INSERT쿼리가 나와서요...
회원가입() 메서드 테스트 시에는 INSERT쿼리가 로그에 찍히지 않는데 중복_회원_예외() 메서드에는 INSERT쿼리가 찍히는 이유가 궁금합니다.
답변 1
8
안녕하세요. saemyung2000님^^ 좋은 질문입니다.
먼저 이 부분을 명확하게 이해하려면 JPA 기본기에 대한 이해가 필요합니다. 지금부터 설명드리는 내용이 좀 어렵다면, 커리큘럼에서 소개드린 것 처럼 우선 활용1편을 쭉 진행하신 다음에 기본편 강의를 들으시는 것을 권장드립니다.
우선 insert 쿼리가 실행이 되더라도 롤백이 되면 DB에 데이터가 남지 않습니다. insert 쿼리가 실행되더라도, 이후에 커밋이 되어야 데이터가 남고, 롤백이 되면 데이터가 남지 않는 것이지요.
그러면 어떤 경우에 로그에 insert 쿼리가 남고, 어떤 경우에는 남지 않을까요?
JPA는 트랜잭션을 커밋하는 시점에 영속성 컨텍스트를 플러시 하는데 이때 내부에서 변경 감지 기능이 발생하고, 새로운 엔티티가 있으면 INSERT 쿼리를 만들어서 DB에 넘깁니다. 그런데 롤백을 실행하면 영속성 컨텍스트를 플러시 하지 않기 때문에 DB에 insert 쿼리를 넘기지 않습니다. 이게 회원가입()의 시나리오에서 롤백으로 처리된 부분이지요.
그런데 중복_회원_예외()는 왜? insert 쿼리가 남았을까요?
그건 바로 영속성 컨텍스트를 플러시 하는 경우가 하나 더 있기 때문입니다! 바로 JPQL을 실행할 때 입니다.
JPQL은 SQL로 DB에서 데이터를 찾기 때문에 우선 영속성 컨텍스트에 있는 내용을 DB에 반영한 다음에 SQL을 실행해야 데이터가 누락되지 않습니다. 그래서 JPA는 JPQL 실행 직전에 자동으로 영속성 컨텍스트를 플러시 합니다.
그래서 비밀은 바로 다음 코드의 findByName에 있습니다.
private void validateDuplicateMember(Member member) {
List<Member> findMembers = memberRepository.findByName(member.getName());
여기 내부에서 JPQL을 실행하기 때문입니다. 중복 회원 가입은 두번째 회원 가입을 할 때 영속성 컨텍스트에 첫번째 회원이 존재합니다. 따라서 이 JPQL이 실행될 때 영속성 컨텍스트에 플러시가 일어나면서 첫번째 회원을 DB에 저장합니다. 그래서 insert 쿼리가 나타나는 것이지요.
내용이 어려우시면 2가지만 기억하시면 됩니다.
1. JPA는 트랜잭션 커밋 시점에 플러시가 발생한다.
2. JPA는 JPQL을 실행할 때 플러시가 발생한다.
감사합니다^^
sdk 설정 오류
0
50
2
오탈자 - @Transactional
0
55
1
src/test/resources 테스트 경로 문제
0
50
1
상품 등록후 H2 db 출력 순서 바꿀 수 있나요?
0
64
1
MemberRepositoryTest 실행오류
0
81
1
boot 4.x >>> trasasction rolled back log & p6spy(영한님, 수업 자료 업데이트 해주시면 감사하겠습니다!!)
1
183
2
강의 마지막 QueryDSL 사용 부분 질문있습니다
1
139
2
클라이언트에서 isbn과 author 수정 요청을 한 경우에 대해 질문드립니다.
0
51
1
도메인 모델 패턴 vs 트랜잭션 스크립트 패턴
0
71
1
기본 생성자
0
60
1
h2 DB 연결시 jdbc url 변경 이유가 궁금합니다.
0
101
1
멤버서비스테스트 부분에서 막힙니다.
0
164
4
실무에서도 EntityManager를 이용해서 많이 작업하는 편일까요?
0
116
1
초반에 h2 다운로드 과정 꼭 필요한가요?
0
119
2
자신 필드에도 get으로 접근하는 이유가 있을까요?
0
114
1
24분 27초 연관관계 편의 메서드 위치
0
113
1
단건 주문만 가능하게 한건 의도한 부분이신가요?
0
109
2
빌드 툴, Gradle
0
59
1
h2연결은 된 것 같은데 엔티티 테이블까지 작성 후 확인해보아도 테이블이 안보입니다
0
77
2
Repository에서 EntityManager 주입 방식 차이
0
90
1
롬복과 사용자 정의 setter 메서드
0
71
1
주문 목록 조회 fetch join 질문드립니다
0
81
1
dirty checking 질문드립니다.
0
82
1
동시성 관련 질문입니다
0
74
1





