toMany 관계에서 where 절
203
작성한 질문수 4
안녕하세요 강사님 jpa 공부중에 궁금한게 있어 질문드립니다.
먼저 OneToMany 양방향 관계인 Team과 Member가 있습니다.
Team을 조회하면서 그중 나이가 10살 이상인 Member만 조회하도록 다음과 같이 코드를 작성했습니다
@Test
void test2() {
QTeam team = QTeam.team;
QMember member = QMember.member;
JPAQueryFactory query = new JPAQueryFactory(em);
List<Team> result = query.selectFrom(team)
.join(team.members, member)
.where(member.age.eq(10))
.fetch();
result.forEach(t -> {
t.getMembers().forEach(m -> {
Assertions.assertThat(m.getAge()).isEqualTo(10);
});
});
}
테스트 코드는 실패를 했습니다. 원인은 where절이 Team을 조회할때만 적용되었고 LazyLoading으로 Member를 가져올 때는 적용되지 않기 때문입니다. 그래서 이를 해결하기 위해 Member에 대한 쿼리를 추가적으로 작성하여 조회를 했습니다.
@Test
void test3() {
QTeam team = QTeam.team;
QMember member = QMember.member;
JPAQueryFactory query = new JPAQueryFactory(em);
List<Team> result = query.selectFrom(team)
.join(team.members, member)
.where(member.age.eq(10))
.fetch();
List<Long> teamIds = result.stream().map(Team::getId).collect(Collectors.toList());
List<Member> members = query.selectFrom(member)
.where(
member.team.id.in(teamIds),
member.age.eq(10)
)
.fetch();
Map<Long, List<Member>> memberMap = members.stream().collect(Collectors.groupingBy(m -> m.getTeam().getId()));
result.forEach(t -> t.setMembers(memberMap.get(t.getId())));
result.forEach(t -> {
t.getMembers().forEach(m -> {
Assertions.assertThat(m.getAge()).isEqualTo(10);
});
});
}
그리고 이 테스트는 통과를 했고 출력값도 원하는 값을 얻게 되었습니다.
여기서 궁금증이 하나 생겼습니다.
fetchjoin의 대상에는 where절을 걸면 안된다 -> fetchjoin의 경우 연관된 모든 엔티티가 존재할것으로 가정하고 사용해야 하기에 객체와 DB의 일관성이 깨지기 때문이다.
를 검색해서 알게 되었는데 제 코드의 경우 fetchjoin을 사용하지 않았습니다. 과정이 어쨋든간에 엔티티 조회를 했고 조회한 엔티티가 온전한 데이터를 모두 갖고 있는게 아닌 필터링된 데이터만 갖고 있기에 위의 fetchjoin처럼 데이터의 일관성이 깨져있는것으로 보이는데 제가 이해한것이 맞는지 궁금합니다.
또한 다음과 같이 엔티티에 where 절을 걸어서 조회하는 케이스도 데이터의 일관성이 깨진다고 볼 수 있는지 궁급합니다.
@Entity
@Getter
@Setter
@NoArgsConstructor(access = AccessLevel.PROTECTED)
@ToString(of = {"id", "username", "age"})
@Where(clause = "age = 10")
public class Member {
@Id
@GeneratedValue
@Column(name = "member_id")
private Long id;
private String username;
private int age;
@ManyToOne(fetch = FetchType.LAZY)
@JoinColumn(name = "team_id")
private Team team;
public Member(String username) {
this(username, 0);
}
public Member(String username, int age) {
this(username, age, null);
}
public Member(String username, int age, Team team) {
this.username = username;
this.age = age;
if (team != null) {
changeTeam(team);
}
}
public void changeTeam(Team team) {
this.team = team;
team.getMembers().add(this);
}
}
답변 1
1
안녕하세요. rkdals213님^^
데이터의 일관성이 깨진다고 표현한 부분은 엔티티를 fetch join으로 조회했을 때 입니다. 엔티티가 아닌 다른 경우는 상관이 없습니다.
아마 이 링크인 것 같네요. https://www.inflearn.com/questions/15876
예제의 경우 처음부터 DTO로 조회하시면 원하시는 결과를 더 편리하게 뽑을 수 있을거에요.
추가로 @Where는 실무에서는 각각 상황에 따라서 다른 쿼리를 실행해야 하기 때문에 권장하지 않습니다. @Where는 이런 문제점 때문에 제가 실무에서 잘 사용하지 않아서 잘 모르겠습니다.
감사합니다.
강의 관련 외 질문입니다.
0
67
2
SpringBoot4 + Hibernate7 모듈 등록 방법 공유
0
92
1
BeanCreationException
0
90
3
Update 후 UpdateMemberResponse 매핑할 때
0
50
1
트랜잭션을 사용 안 할 때 커넥션은 언제 가져오나요?
0
101
2
페이징 + 검색조건 관련해서 질문드립니다.
0
70
1
Query Dsl Q파일 질문입니다.
0
84
1
루트 쿼리라는것은
0
59
1
메서드를 분리하는 기준
0
63
1
findAllWithMemberDelivery 메서드 질문드립니다.
0
110
3
연관관계 매핑을 안 쓸 경우, 사용해야 하는 전략
0
86
2
fetch join과 영속화와 OSIV의 관계
0
86
2
Distinct 사용 전 결과에 대한 의문
0
115
2
레포지토리 계층에서의 트랜잭션에 대한 의문
0
59
1
영속성 컨텍스트 생명주기의 신기한 부분이 있습니다.
0
78
2
dto 필드 속 엔티티 여부
0
60
1
뷰템플릿 사용 시
0
77
2
Result 클래스 관련 질문
0
56
1
@PostConstruct 프록시 관련 질문드립니다
0
86
1
DTO 대신 Form 사용은 안되나요?
0
138
1
OSIV ON 상태일 때
0
96
1
fetch join VS fetch join 페이징 궁금증
0
185
2
양방향 연관관계 알아보는 법?
0
106
1
16강 17강 간단 정리 이게 맞을까요 ?
0
165
2





