작성
·
159
0
이번 강의 보다가 궁금한게 생겼는데
======================================
Member member = em.find(Member.class, 0l);
em.detach(member);
member.setName("zzzz"); // 기존 name은 "xxxxx"
em.merge(member);
tx.commit();
======================================
1) 위의 코드를 실행했을 때 왜 변경사항이 반영될까요?? 1차 캐시에서 준영속 상태도 변경사항을 따로 관리해준다고 이해해도 될까요??
--------------------------------------------------
2) em.persist(member); 를 commit 바로 이전에 넣었을 때는 변경사항이 반영이 안되더라구요
1)번을 봤을 때 merge(member) 후 1차 캐쉬에 name이 "zzzz"인 member의 정보가 있고, persist(member)를 했을 때 member객체의 name이 "zzzz"이기 때문에
DB에는 ID가 "0"인 멤버의 NAME이 "zzzz"일 것이라 생각했는데 DB를 보니 반영되지 않았더라구요 (기존 id==0인 member의 name은 "xxxxx"이고 결과도 "xxxxx")
--------------------------------------------------
3)
1)에서 콘솔창에 찍힌 sql은 select, select, insert
2)에서 콘솔창에 찍힌 sql은 select, select만 있었습니다
왜 펑션을 더 많이 호출했던 1)의 sql이 더 많이 찍혔을 까요..?
--------------------------------------------------
혼자 오래 고민하다 왜 그런지 감이 안잡혀 질문 드립니다.ㅜㅜ
- 좋은 강의 만들어주셔서 감사합니다. 너무 잘 보고 있어요!!
답변 3
1
1
안녕하세요. 남상민님 좋은 질문입니다.
merge는 준영속 상태의 엔티티로 영속 상태의 엔티티를 새로 만들어 냅니다. 다음과 같은 코드가 있을 때
Member mergeMember = em.merge(member);
member는 여전히 준영속 상태이지만 그 결과로 나온 mergedMember는 영속 상태가 됩니다.
em.merge를 호출하면 다음과 같은 동작이 일어납니다.
1. 파라미터로 넘긴 member의 id로 DB를 조회한 다음 영속 상태의 mergeMember(xxxx)를 생성합니다.
2. mergeMember에 파라미터로 넘긴 member(zzzz)의 모든 값을 밀어 넣습니다.
3. newMember(zzzz)를 반환합니다.
결과적으로 newMember의 값이 xxxx -> zzzz로 변하게 되므로 변경감지에서 db에 update가 발생합니다.
이 내용을 이해하시면 1,2,3,이 풀리실 꺼에요^^
merge는 이론을 학습해도 이해하기가 쉽지 않아서, 제가 기본편에 일부러 설명을 하지 않고, 활용 1편에서 실제 데이터를 수정하는 부분에서 설명을 드립니다^^ (활용1 - 변경 감지와 병합(merge) 참고)
혹시 merge에 대해 더 자세한 이론 내용을 학습하고 싶으시면 JPA 책 3.6.5 병: merge()를 참고해주세요 :)
감사합니다.
0