작성
·
228
1
@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
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님
전체 프로젝트를 압축해서 구글 드라이브로 공유해서 링크를 남겨주세요.
구글 드라이브 업로드 방법은 다음을 참고해주세요.
주의: 업로드시 링크에 있는 권한 문제 꼭 확인해주세요
추가로 다음 내용도 코멘트 부탁드립니다.
1. 실행 방법을 알려주세요.
2. 어떻게 문제를 확인할 수 있는지 자세한 설명을 남겨주세요.
감사합니다.