JPA
[JPA] Fetch - 즉시 로딩, 지연 로딩
mangdo
2021. 7. 14. 22:26
⌚ 지연 로딩(FetchType.LAZY)
@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;
..
}
- Entity를 처음 생성할때
- 실제로 객체를 조회할때
Team team = member.getTeam();
team.getName(); // 실제 team을 사용하는 시점에 프록시 초기화 이때 쿼리가 나간다.
단순히 getTeam()으로 가져올 때가 아니고 team에 있는 무언가를 실제 사용할때 초기화된다!
⌚ 즉시 로딩(FetchType.EAGER)
@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;
..
}
: Member를 로딩할때 Team까지 조인해서 가져온다.
⌚ 지연 로딩 VS 즉시 로딩
가급적 지연로딩만 사용해야한다!!! 특히 실무에서와 같이 테이블이 복잡하게 얽혀있는 경우에는 지연로딩을 사용해야한다. 그이유는 다음과 같다.
1. 즉시로딩을 사용하면 예상하지 못한 SQL이 발생한다.
-> 사실 이 이슈는 내가 진행하던 프로젝트에서 발생했었다. (실제 프로젝트 이슈)
2. 즉시로딩은 JPQL에서 N+1 문제를 일으킨다.
⌚ 즉시 로딩의 N+1 문제
List<Member> members = em.createQuery("select m from Member m", Member.class).getResultList();
// JPQL은 그대로 SQL로 번역이된다.
// 즉 Member만 일단 SELECT한다.
// Member 다가져왔는데 EARGER러구...? Team을 다시 가져올게!
// SQL 1 : select * from Member
// SQL N : select * from Team where TEAM_ID = xxx
쿼리를 하나를 날렸는데 N개가 딸려올 수 있다. 실무에서는 JPQL을 많이 사용하고 이러한 문제가 많이 발생한다고 한다.
이 문제를 해결하기 위해서는 모든 연관관계를 일단 지연로딩으로 해야한다.
그러고 정말 필요해서 같이 조인해야한다면, Fetchjoin 이라는 동적으로 원하는 쿼리들을 한번에 가져올 수 있다.
List<Member> members = em.createQuery("select m from Member m join fetch m.team", Member.class).getResultList();
⌚ fetch 기본값
1. @ManyToOne, @OneToOne은 기본이 즉시 로딩-> LAZY로 설정
2. @OneToMany, @ManyToMany는 기본이 지연 로딩