JPA

[Querydsl] 튜플이나 DTO로 결과 반환하기

mangdo 2021. 8. 5. 00:55

 

프로젝션 : select 대상지정하는 일

프로젝션 대상이 두개 이상이라면 튜플이나 DTO로 조회해야한다.

 

🌱 튜플 사용하기

com.querydsl.core.Tuple를 사용하고 있다.

때문에 Repository 계층을 넘어서 Service나 Controller계층에 넘어가는 것은 좋지않은 설계다.
Service나 Controller계층에 넘어갈때는 DTO로 넘어가는 것이 좋다고 생각한다.

List<Tuple> result = queryFactory
                        .select(member.username, member.age)
                        .from(member)
                        .fetch();

for (Tuple tuple : result) {
    String username = tuple.get(member.username);
    Integer age = tuple.get(member.age);
    
    System.out.println("username=" + username);
    System.out.println("age=" + age);
}

 

🌱 DTO 사용하기

DTO를 사용하려면 다음 3가지 방법 지원한다.

  • 프로퍼티 접근
  • 필드 직접 접근
  • 생성자 사용

 

1. Bean() -> getter, setter, 디폴트 생성자 필요

List<MemberDto> result = queryFactory
                          .select(Projections.bean(MemberDto.class,
                                        member.username,
                                        member.age))
                          .from(member)
                          .fetch();

2. 필드 직접 접근 -> getter, setter필요없음. 바로주입

List<MemberDto> result = queryFactory
                        .select(Projections.fields(MemberDto.class,
                                    member.username,
                                    member.age))
                        .from(member)
                        .fetch();

3. 생성자 사용

List<MemberDto> result = queryFactory
                            .select(Projections.constructor(MemberDto.class,
                                      member.username,
                                      member.age))
                            .from(member)
                            .fetch();

 

번외 ) DTO의 이름이 다르다면?

ExpressionUtils.as(source,alias)-> 필드나, 서브 쿼리에 별칭 적용

List<UserDto> fetch = queryFactory
                        .select(Projections.fields(UserDto.class,
                                member.username.as("name"),
                                ExpressionUtils.as(
                                    JPAExpressions
                                    .select(memberSub.age.max())
                                    .from(memberSub), "age")
                                ))
                        .from(member)
                        .fetch();

 

4. @QueryProjection

DTO 생성자에 @QueryProjection을 붙여주면 DTO도 Q파일로 생성된다.

import com.querydsl.core.annotations.QueryProjection;

public class MemberDto {

    private String username;

    private int age;

    public MemberDto() {
    }

    @QueryProjection
    public MemberDto(String username, int age) {
        this.username = username;
        this.age = age;
    }
}

생성된 Q파일을 사용하면된다.

queryFactory
    .select(new QMemberDto(member.username, member.age))
    .from(member)
    .fetch();

-> 이 방법의 좋은점은 컴파일오류로 많은 것을 잡아낼 수 있다는 것이다.

-> 컴파일 시점에 타입 체크, 파라미터 갯수체크 등 가능하다.

-> 단, memberDto가 Querydsl에 대한 의존성이 생긴다는 단점이 있다.

    특히 DTO는 Service계층, Controller계층등 여러 계층을 넘어다니는 객체임으로 아키텍쳐 전반적으로 Querydsl에 대한 의존성이 생기는 것이 큰 단점이 생길 수 있다.