지연 로딩을 사용한 프록시로 조회
@Entity
public class Member {
@Id
@GeneratedValue
private Long id;
@Column(name = "USERNAME")
private String name;
@ManyToOne(fetch = FetchType.LAZY) //**
@JoinColumn(name = "TEAM_ID")
private Team team;
..
}
지연 로딩 동작 방식
- 초기화 전
Member member = em.find(Member.class, 1L);
- 초기화 후
Team team = member.getTeam();
team.getName(); // 실제 team을 사용하는 시점에 초기화(DB 조회)
즉시로딩
@Entity
public class Member {
@Id
@GeneratedValue
private Long id;
@Column(name = "USERNAME")
private String name;
@ManyToOne(fetch = FetchType.EAGER) //**
@JoinColumn(name = "TEAM_ID")
private Team team;
..
}
즉시로딩 동작 방식
즉시로딩(EAGER)은 Member 조회시 항상 Team도 같이 조회
※ JPA 구현체는 가능하면 조인을 사용해서 SQL 한번에 함께 조회
즉시로딩 주의점
• 가급적 지연 로딩만 사용(특히 실무에서)
• 즉시 로딩을 적용하면 예상하지 못한 SQL이 발생
• 즉시 로딩은 JPQL에서 N+1 문제를 일으킨다(Team을 사용하지 않을때도 항상 같이 조회하기 때문)
• @ManyToOne, @OneToOne은 기본이 즉시 로딩 -> LAZY로 설정
• @OneToMany, @ManyToMany는 기본이 지연 로딩
실무 주의점
• 모든 연관관계에 지연 로딩을 사용해라!
• 실무에서 즉시 로딩을 사용하지 마라!
• JPQL fetch 조인이나, 엔티티 그래프 기능을 사용해라!
• 즉시 로딩은 상상하지 못한 쿼리가 나간다.
'JAVA > JPA' 카테고리의 다른 글
(JPA) 기본값과 임베디드 타입 (0) | 2021.07.06 |
---|---|
(JPA) 영속성 전이, 고아 객체 (0) | 2021.07.06 |
(JPA) Proxy(프록시) (0) | 2021.07.04 |
(JPA) @MappedSuperclass (0) | 2021.07.04 |
(JPA) 상속 관계 매핑 (0) | 2021.07.03 |