인프런 커뮤니티 질문&답변

Jeonghee Lee님의 프로필 이미지

작성한 질문수

자바 ORM 표준 JPA 프로그래밍 - 기본편

영속성 컨텍스트 1

비영속 준영속 상태에 대해 궁금한 것이 있습니다.

작성

·

228

1

[질문 템플릿]
1. 강의 내용과 관련된 질문인가요? 예
2. 인프런의 질문 게시판과 자주 하는 질문에 없는 내용인가요? 예
3. 질문 잘하기 메뉴얼을 읽어보셨나요? 예

[질문 내용]
 
개인적으로 궁금한 내용이 있습니다. 공부하면서 궁금한 점이 있어서 아래와 같은 서비스를 만들고 controller 에서 이 서비스의 함수를 호출하게 했습니다. 제 생각으로는 em.detach를 호출해 영속성 컨텍스트에서 관리되지 않고 @transactional 때문에 commit 전이라 db에 데이터가 들어가면 안된다고 생각했습니다. 그러나 확인해보니 db에 데이터가 들어갔습니다. 혹시나 해서 em.clear()도 해봤는데 동일하게 데이터가 들어갔습니다. detach, clear 했음에도 데이터가 insert되는 이유가 궁금해서 질문드립니다.
 
@Service
@RequiredArgsConstructor
public class MemberServiceImpl implements MemberService {

private final EntityManager em;

@Override
@Transactional
public void test() {
Member member = new Member();
em.persist(member);
em.detach(member);
}
}
 

답변 3

1

김영한님의 프로필 이미지
김영한
지식공유자

안녕하세요. Jeonghee Lee님

디테일하게 들어가면 2가지 경우로 나눌 수 있습니다.

1. auto increment 같은 전략을 사용하기 위해 IDENTITY 방법을 쓰는 경우

public class Member {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;

이 경우에는 em.persist()로 엔티티를 저장하는 시점에 DB에 INSERT 쿼리를 실행하게 됩니다. 그렇게 해야 ID 값을 획득할 수 있기 때문이지요. JPA에는 영속화 시점에 항상 @Id의 값이 필수로 필요합니다.

이미 INSERT 쿼리가 실행되어 버렸기 때문에 이후에 커밋이 일어나게 되면 DB에 반영됩니다.

2. 나머지 경우

@Id만 사용해서 ID를 직접 지정하거나 또는 GenerationType.SEQUENCE, GenerationType.TABLE를 사용하는 경우

이런 경우에는 em.persist()를 호출해도 식별자를 별도의 방법으로 조회하기 때문에 DB에 INSERT 쿼리를 바로 보내지 않습니다. 따라서 이 경우에는 detach를 사용하면 영속성 컨텍스트에서 해당 엔티티가 제거되기 때문에 INSERT 쿼리가 나가지 않습니다. 이후에 커밋이 일어나도 DB에 데이터가 저장되지 않습니다.

참고로 이런 내용들이 복잡해보여도 크게 문제가 되지 않는 이유는 em.detach()를 실무에서는 거의 사용할 일이 없기 때문입니다. 더욱이 em.persist() 직후에 em.detach()를 사용하는 일은 없다고 생각하시면 됩니다.

감사합니다.

0

Jeonghee Lee님의 프로필 이미지
Jeonghee Lee
질문자

https://drive.google.com/file/d/1-WuKsSWGla0LNqD55XNkY3R8fbPhAVwH/view?usp=sharing

 

1. 실행 방법을 알려주세요.

- h2 database url에 맞게 설정한 뒤 application run

- postman으로 http://localhost:8080/test (GET) 요청 send

 

2. 어떻게 문제를 확인할 수 있는지 자세한 설명을 남겨주세요.

- http://localhost:8080/h2-console 에 들어가서 Member table select

- postman으로 http://localhost:8080/test (GET) 요청 send 후에 h2-console에서 member 테이블을 select 하면 member 데이터가 들어가 있는 것을 확인할 수 있습니다.

- 저는 persist 후에 em.detach(member)를 해서 영속성 컨텍스트가 해당 엔티티를 관리하지 않기 떄문에 실제로는 member데이터가 db에 insert되면 안된다고 생각했습니다.

- em.detach() 대신 em.clear()을 사용해도 데이터가 들어갔습니다.

- member 데이터가 em.detach() / em.clear()을 해서 영속성 컨텍스트에서 관리가 되지 않음에도 데이터가 insert되는 이유를 알고 싶습니다.

 

0

김영한님의 프로필 이미지
김영한
지식공유자

안녕하세요. Jeonghee Lee님

전체 프로젝트를 압축해서 구글 드라이브로 공유해서 링크를 남겨주세요.

구글 드라이브 업로드 방법은 다음을 참고해주세요.

https://bit.ly/3fX6ygx

 

주의: 업로드시 링크에 있는 권한 문제 꼭 확인해주세요

 

추가로 다음 내용도 코멘트 부탁드립니다.

1. 실행 방법을 알려주세요.

2. 어떻게 문제를 확인할 수 있는지 자세한 설명을 남겨주세요.

감사합니다.