인프런 영문 브랜드 로고
인프런 영문 브랜드 로고

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

김연규님의 프로필 이미지
김연규

작성한 질문수

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

단방향 연관관계

JPA에서 DB에 쿼리가 나가는 것 질문

작성

·

334

0

안녕하세요 재밌게, 듣고 공부하고 있습니다. 감사합니다. 

다름이 아니라, 아래 코드에서 제 생각에는 change TEAM할 때, DB에 쿼리가 안나가더라구요??  teamB에 대한 내용을 DB에서 가져올 것으로 생각했거든요. 왜냐하면 영속성 컨텍스트에 없으니까 DB에서 가져와서 1차캐시에 넣는다 라고 생각했습니다. 

하지만 다시 생각해보니, 위에 teamB 객체를 이미 만들어두었기 때문에 이를 changeTeam 할때 넣으니까 쿼리가 DB에 안나가는 것 같습니다. 

 

그러면 commit 할때, update를 치잔아요??  그때, MEMBER 테이블에 변경된 TEAM 외래키를 넣을 텐데, 이때 JPA에서는 teamB의 키를 어떻게 아나요?? 영속성 컨텍스트에는 일단 teamB에 대한 내용은 없고, java 메모리에 teamB 있는 걸로 setTeam 하여 변경했는데, 업데이트 치려면 DB의 teamB의 키를 가져와야 할 텐데, 쿼리 나가는 것 보니 업데이트만 치더라구요... 

 

글이 길어졌네요... 답변 감사합니다! 

// 저장
Team teamA = new Team();
teamA.setName("TeamA");
em.persist(teamA);

// 저장
Team teamB = new Team();
teamB.setName("TeamB");
em.persist(teamB);

// 저장
Team teamC = new Team();
teamC.setName("TeamB");
em.persist(teamC);



Member member = new Member();
member.setName("member1");
member.setTeam(teamA);
em.persist(member);

em.flush();
em.clear();

// 조회
System.out.println("======find Member =====");
Member findMember = em.find(Member.class, member.getId());
System.out.println("======find Team Name ==== ");
System.out.println("findTeam = " + findMember.getTeam().getName());

// 팀 변경
System.out.println("======change TEAM =====");
findMember.setTeam(teamB);
System.out.println("======find Team Name ==== ");
System.out.println("findTeam = " + findMember.getTeam().getName());



tx.commit(); // 이때 DB에 query가 날라감






======find Member ===== Hibernate: select member0_.MEMBER_ID as MEMBER_I1_0_0_, member0_.USERNAME as USERNAME2_0_0_, member0_.TEAM_ID as TEAM_ID3_0_0_ from Member member0_ where member0_.MEMBER_ID=? ======find Team Name ==== Hibernate: select team0_.TEAM_ID as TEAM_ID1_1_0_, team0_.name as name2_1_0_ from Team team0_ where team0_.TEAM_ID=? findTeam = TeamA ======change TEAM ===== ======find Team Name ==== findTeam = TeamB Hibernate: /* update hellojpa.Member */ update Member set USERNAME=?, TEAM_ID=? where MEMBER_ID=?

답변 1

0

안녕하세요, 김연규 님. 공식 서포터즈 codesweaver 입니다.
.
최초 teabB를 생성하신 이후에 em.persist(teamB); 하신 순간 teamB에는 디비에서 얻어온 아이디 값이 들어오게 됩니다.

.

findMember.setTeam(teamB);. 했을때 쿼리가 날아가지 않는 부분은 말씀하신것처럼, 더티체킹은 트랜잭션이 끝나는 순간 발생하기 때문이지요.

.

트랜젹션이 끝나 실제로 업데이트를 하려 할 때, teamB 는 이미 고유한 아이디를 가지고 있으므로 문제없이 디비에 입력이 되는것입니다.

.
감사합니다.

김연규님의 프로필 이미지
김연규

작성한 질문수

질문하기