Written by
Booki
on
on
[JPA] API 성능 최적화 하기 (1) - X to One 연관 관계
이 글은 김영한 님의 실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화 편을 보고 남기는 정리글 입니다.
X to One 연관 관계의 성능 최적화를 하기위해 먼저 알아야 할 것은 즉시(Eager) 로딩과 지연(Lazy) 로딩의 차이이다.
Member.java
@Entity
@Getter
@Setter
public class Member {
@Id
@GeneratedValue
private Long id;
private String name;
@ManyToOne(fetch = FetchType.EAGER)
private Team team;
}
Team.java
@Entity
@Getter
@Setter
public class Team {
@Id
@GeneratedValue
private Long id;
private String name;
@OneToMany(mappedBy = "team")
private List<Member> members = new ArrayList<>();
}
간단하게 두 엔티티의 연관 관계를 만들어주고, FetchType.EAGER
로 설정을 해두었다.
간단한 엔티티 테스트 코드를 통해 쿼리문을 확인해보도록 하자.
MemberTest.java
@SpringBootTest
@Transactional
class MemberTest {
@PersistenceContext
EntityManager em;
@Test
public void testEntity() {
Team teamA = new Team("teamA");
em.persist(teamA);
Member member1 = new Member("member1", 10, teamA);
em.persist(member1);
// 초기화
em.flush();
em.clear();
// 확인
em.find(Member.class, member1.getId());
}
}
결과
Hibernate:
select
m1_0.member_id,
m1_0.age,
t1_0.team_id,
t1_0.name,
m1_0.username
from
member m1_0
left join
team t1_0
on t1_0.team_id=m1_0.team_id
where
m1_0.member_id=?
FetchType.EAGER
로 설정되어 있을 때는 이와 같이 member만 조회했음에도 team까지 함께 join되어 결과 값을 가져온다.
다음은 FetchType.LAZY
로 설정했을 때의 결과 값이다.
결과
Hibernate:
select
m1_0.member_id,
m1_0.age,
m1_0.team_id,
m1_0.username
from
member m1_0
where
m1_0.member_id=?
FetchType.LAZY
로 설정되어 있을 때는 team까지는 가져오지 않고 조회한 member만 가져오게 된다.
JPA에서 X to One 연관 관계의 기본 값은 FetchType.EAGER
로 설정되어 있다. 이로 인해 필요치 않은 데이터가 같이 조회되며 N+1 문제를 일으킬 수 있다.
지연 로딩을 피하기 위해 FetchType.EAGER
로 설정하는 것은 성능 튜닝을 어렵게 만드는 지름길이다.
성능 최적화가 필요한 경우에는 Fetch Join
을 사용하면 된다. 다음 편에서는 이 fetch join을 사용하여 성능 최적화하는 방법을 알아보도록 하자.