inflearn logo
강의

강의

N
챌린지

챌린지

멘토링

멘토링

N
클립

클립

로드맵

로드맵

지식공유

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

페치 조인 1 - 기본

JOIN FETCH와 1차 캐시 질문드립니다.

556

jaewkim

작성한 질문수 5

0

JOIN FETCH를 이용하여 쿼리를 작성하고 테스트 해보던 중에 이해가지 않는 동작이 발견되어 질문드려요!

JOIN FETCH로 채우려는 필드가, 1차 캐시에 엔티티가 이미 존재할 때에, 해당 필드가 null 이 되는 현상입니다.

오히려 영속성 컨텍스트를 비워주면 프록시를 가져오는 것 같습니다... 어떻게 동작하는 건지 질문 드립니다!


테스트 메서드를 간단하게 작성하며 생성 및 조회를 하나의 트랜잭션에서 진행했습니다.

엔티티는 간략하게 다음과 같습니다.

@Entity

public class Collection {

  @Id
  @GeneratedValue
  private Long id;

  @OneToMany(fetch = FetchType.LAZY, mappedBy = "collection", cascade = CascadeType.ALL, orphanRemoval = true)
  private List<ReviewInCollection> reviews = new ArrayList<>();
   
  // ...
}

@Entity
public class ReviewInCollection {

  @Id
  @ManyToOne(fetch = FetchType.LAZY)
  @JoinColumn(name = "COLL_ID")
  private Collection collection;

  @Id
  @ManyToOne(fetch = FetchType.LAZY)
  @JoinColumn(name = "REV_ID")
  private Review review;

  // ...
}

@Entity
public class Review {

  @Id
  @GeneratedValue
  @Column(name = "REV_ID")
  private Long id;

  @ManyToOne(fetch = FetchType.LAZY)
  @JoinColumn(name = "PLACE_ID", nullable = false)
  private Place place;

  @ManyToOne(fetch = FetchType.LAZY)
  @JoinColumn(name = "MEMBER_ID", nullable = false)
  private Member member;
 
  // ...
}

 

문제가 발생한 코드입니다.

  @Test
  void updateReviewsTest() {
    final SessionImplementor session = em.unwrap(SessionImplementor.class);
    final PersistenceContext pc = session.getPersistenceContext();

    // mock up

    //!=========================================================================================!//

    em.flush();
    // em.clear();
    Collection coll = em.createQuery(
                            "select c from Collection c " 
                            + "join fetch c.reviews cr " 
                            + "join fetch cr.review r where c.id = :collId ",
                            Collection.class)
                        .setParameter("collId", 1L)
                        .getSingleResult();

    coll.getReviews() // ReviewInCollection
        .stream()
        .map(cr -> cr.getReview() // Review → NPE
                     .getId()
             )
        .toList();
   }
Cannot invoke "com._2cha.demo.review.domain.Review.getId()" because the return value of "com._2cha.demo.collection.domain.ReviewInCollection.getReview()" is null
java.lang.NullPointerException: Cannot invoke "com._2cha.demo.review.domain.Review.getId()" because the return value of "com._2cha.demo.collection.domain.ReviewInCollection.getReview()" is null

JOIN FETCH 를 통해

Collection.reviews
ReviewInCollection.review

를 채워 오고자 했는데,
Collection → for each ReveiwInCollection → review 가 null 로 채워지게 됩니다.


> 쿼리 이전의 영속성 컨텍스트입니다.


> 쿼리 이후 영속성 컨텍스트입니다.

 

주석 처리한 em.clear() 를 실행하면,  null이 아닌 프록시로 가져와서 LAZY 방식으로 Review를 채웁니다.

LAZY 또한 예상한 동작이 아니긴 하나, 일단 왜 null이 들어가는지가 가장 궁금합니다... 영속성 컨텍스트와 JOIN FETCH의 충돌이 있는 걸까요?

java jpa

답변 1

0

김영한

안녕하세요. jaewkim님

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

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

https://bit.ly/3fX6ygx

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

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

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

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

감사합니다.

 

벌크연산에서 member.getAge 호출 시 영속성 컨텍스트에서 데이터를 가져오는건가요?

0

28

2

inheritance startegy 선택시 고려사항

0

22

1

Entity 동등성 비교

0

21

1

실무 조언 관련 질문입니다.

0

47

1

H2데이터베이스 파일 생성

0

56

2

서브쿼리 강의에서 ALL 예시 관련 질문드립니다.

0

53

2

수정또는 삭제시 영속성 엔티티에 값이 무조건 있어야 하나요?

0

52

1

JPQL 메소드와 락

0

55

1

Delivery @OneToOne

0

60

1

17강 4~5분대 테이블 값 조회가 안됩니다.

0

94

2

UnsupportedOperationException 발생

0

86

3

H2 Database 연결이 안됩니다.

0

95

2

연관관계 매핑 질문드립니다.

0

85

2

h2데이터베이스 실행오류

0

108

2

persistence.xml

0

108

2

양방향 연관관계에서 연관관계의 주인(mappedBy)을 왜 꼭 정해야 하나요?

0

80

1

영속성 컨텍스트

0

66

1

JPA 프록시

0

96

1

Native Query와 MyBatis

0

70

1

영속성 컨텍스트는 어떤 메모리에 저장되는건가요?

0

87

1

임베디드 타입 예시 코드 관련 질문

0

115

3

명시적 조인에서 별칭을 주면 왜 객체에 접근할 수 있나요

0

95

3

인텔리제이 패키지 커서 단축키 질문

0

108

2

혹시 현재는 ID 데이터 타입이 String이면 안되나요?

0

145

1