@ManyToOne FetchType=EAGER로 조회 시 N+1문제 발생
248
작성한 질문수 1
안녕하세요 기선님
JPA 강좌 정말 잘 보고있습니다!
@ManyToOne 관계 매핑한 엔티티를 EAGER 타입으로 조회를 할때요
어떤 경우에는 조인으로 두 엔티티 정보를 한번에 쿼리하는 반면에,
각각의 엔티티를 따로 쿼리하는 경우가 있어서 EAGER 조회에서도 N+1문제가 발생하더라고요.
어떤 경우엔 조인이 되고 어떤 경우에 각각 쿼리가 발생하는지
알려주실 수 있을까요?
( 지금 껏 테스트로 추측하기로는 id 로 조회할 때는 조인 쿼리가 발생하고,
그외에는 별개 쿼리가발생하는 것 같아요)
코드랑 쿼리 결과도 첨부드릴게요
@Entity
public class Post {
@Id @GeneratedValue
private Long id;
private String name;
public Post() {}
public Post(String name) {
this.name = name;
}
}
@Entity
public class Comment {
@Id @GeneratedValue
private Long id;
private String title;
@ManyToOne
private Post post;
public Comment() {}
public Comment(String title, Post post) {
this.title = title;
this.post = post;
}
}
@Component
public class JpaRunner implements ApplicationRunner {
@Autowired
PostRepository postRepository;
@Autowired
CommentRepository commentRepository;
@Transactional
@Override
public void run(ApplicationArguments args) {
//아래 데이터가 입력된 상황
// Post post1 = new Post("A");
// Post post2 = new Post("B");
// Post post3 = new Post("C");
// postRepository.saveAll(Arrays.asList(post1, post2, post3));
// postRepository.flush();
//
// Comment comment1 = new Comment("a", post1);
// Comment comment2 = new Comment("a", post2);
// Comment comment3 = new Comment("a", post3);
//
// commentRepository.saveAll(Arrays.asList(comment1, comment2, comment3));
// commentRepository.flush();
commentRepository.findById(1L);
commentRepository.findByTitle("a");
}
}
실행된 쿼리
Hibernate:
select
comment0_.id as id1_0_0_,
comment0_.post_id as post_id3_0_0_,
comment0_.title as title2_0_0_,
post1_.id as id1_1_1_,
post1_.name as name2_1_1_
from
comment comment0_
left outer join
post post1_
on comment0_.post_id=post1_.id
where
comment0_.id=?
2018-09-18 02:32:41.805 INFO 5078 --- [ main] o.h.h.i.QueryTranslatorFactoryInitiator : HHH000397: Using ASTQueryTranslatorFactory
Hibernate:
select
comment0_.id as id1_0_,
comment0_.post_id as post_id3_0_,
comment0_.title as title2_0_
from
comment comment0_
where
comment0_.title=?
Hibernate:
select
post0_.id as id1_1_0_,
post0_.name as name2_1_0_
from
post post0_
where
post0_.id=?
Hibernate:
select
post0_.id as id1_1_0_,
post0_.name as name2_1_0_
from
post post0_
where
post0_.id=?
Hibernate:
select
post0_.id as id1_1_0_,
post0_.name as name2_1_0_
from
post post0_
where
post0_.id=?
답변 1
1
좋은 질문 주셔서 감사합니다. 추측하신대로 id로 조회하는 것을 제외하고 기본 패치 모드가 적용되지 않습니다. 따라서 그 부분은 직접 컨트롤 해주시는게 좋습니다. 가령, 뒤에서 학습하실 EntityGraph라는 기능을 사용해서 해당 findByTitle도 Post 데이터를 eager 패치할 수 있습니다.
@EntityGraph(attributePaths = "post")List<Comment> findByTitle(String title);
이렇게 코드랑 로그까지 올려주신 성의에 다시 한번 감사드립니다.
spring boot 2.7.13-SNAPSHOT trace 소문자 로그 안나옴
0
533
1
<스프링 데이터 Common: 기본 리포지토리 커스터마이징> 에 대한 질문
0
392
1
comment table에서 저장될떄 왜 id값이 2부터저장이되는건가요?
0
407
1
@EnableJpaRepositories 설정을 스프링부트가 어디에서 자동설정하나요?
0
451
0
PersistenceContext 관련 질문드립니다.
0
335
1
지금(Eager), 나중에(Lazy)의 의미를 모르겠습니다
0
338
1
transaction 구간이 길어질 경우의 처리방법 문의드립니다.
0
905
1
docker postgres
0
293
1
Multiple DataSource 사용 시 transaction 관련 질문 드립니다.
0
2908
1
entity 중 null이 아닌 필드만 update 할 방법이 있을까요?
0
1191
1
Eager 모드일 경우, join을 inner join으로 바꾸는 법이 있을까요?
0
385
1
엔티티를 상속받는 DTO가 일반적인가요?
1
1847
1
커스텀 타입 클래스를 String 타입 처럼 이용해 쿼리하는 방법에 대해 질문하고 싶습니다.
0
339
1
연관관계 매핑 어떤식으로 해야될지 감이 안잡힙니다.
0
566
4
EntityManager 주입시 Annotation관련 질문드립니다.
0
565
1
클래스 기반 프로젝션 사용 관련 질문
0
560
1
save 메서드 질문드립니다.
0
258
1
복잡한 통계쿼리도 JPA로 가능한가요?
2
5593
1
find 와 get의 차이가 무엇인가요?
0
890
1
실무에서 JPA 할 때 FK로 개발할때 연관관계를 꼭 맺어주어야 하나요?
0
999
1
\dt Did not find any relations.
0
481
1
소스코드는 어디서 볼 수 있을까요?
1
287
1
table 생성과 select 문에 대한 질문
0
174
1
스프링 데이터 RepositoryTest 관련 질문
0
2173
2





