spring/JPA

(JPA) 즉시로딩과 지연로딩

ri5 2021. 7. 4. 18:05

지연 로딩을 사용한 프록시로 조회


 @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 조인이나, 엔티티 그래프 기능을 사용해라!

• 즉시 로딩은 상상하지 못한 쿼리가 나간다.