Note-Taking / / 2024. 9. 20. 11:10

JPQL 핵심 정리

JDBC 템플릿에서는 바인딩?로 지정되지만, JPQL에서는 바인딩이 이름 기반이나 위치 기반으로 이루어져.

 

1. 이름 기반 바인딩 (Named Parameters):

 

JPQL에서 가장 많이 사용하는 방식으로, 이름 기반 바인딩은 파라미터를 :와 함께 이름을 지정해 사용해. 이 방식은 가독성이 높고, 파라미터의 순서와 무관하게 이름으로 바인딩할 수 있어.

 

사용 예시:

String jpql = "SELECT m FROM Member m WHERE m.name = :name";
TypedQuery<Member> query = em.createQuery(jpql, Member.class);
query.setParameter("name", "John");
List<Member> result = query.getResultList();

여기서 :name이 이름 기반의 바인딩 파라미터야. setParameter("name", "John")으로 값을 바인딩해.

 

2. 위치 기반 바인딩 (Positional Parameters):

 

JPQL에서 ?와 함께 사용되는 위치 기반 바인딩도 가능해. JDBC 템플릿처럼 인덱스로 파라미터를 바인딩해.

 

사용 예시:

String jpql = "SELECT m FROM Member m WHERE m.name = ?1";
TypedQuery<Member> query = em.createQuery(jpql, Member.class);
query.setParameter(1, "John");
List<Member> result = query.getResultList();

여기서는 ?1, ?2 등으로 파라미터의 위치를 지정할 수 있어.

 

정리

 

JDBC 템플릿: ?로 파라미터를 바인딩.

JPQL: 이름 기반(:name) 또는 위치 기반(?1) 바인딩을 사용.

 

이름 기반 바인딩이 가독성이 좋고 유지보수하기 쉬워서 더 자주 사용돼.

 

JPQL에서의 프로젝션은 쿼리 결과로 반환할 데이터를 지정하는 방식으로, 쿼리의 결과를 엔티티 전체가 아닌, 일부 필드나 특정 데이터 구조로 반환하는 걸 말해. 더 구체적으로 설명해 볼게.

 

1. 프로젝션의 필요성

 

프로젝션은 데이터베이스 쿼리의 성능을 최적화하고, 필요한 데이터만을 선택적으로 가져오게끔 해. 일반적으로 엔티티 전체를 가져오지 않고, 필요한 필드나 DTO로 쿼리 결과를 제한하여 불필요한 메모리 소비를 줄일 수 있어. 예를 들어, 전체 Member 엔티티가 아닌, 이름나이 필드만 필요하다면 필드 프로젝션을 사용해서 성능을 개선할 수 있지.

 

2. 프로젝션의 종류 (세부적인 설명)

 

2.1 엔티티 프로젝션

 

엔티티 전체를 가져오는 방식으로, 기본적으로 JPQL에서 사용하는 프로젝션이야. JPQL에서 엔티티의 모든 필드를 조회할 때 주로 사용돼.

 

예시:

SELECT m FROM Member m

 

이 쿼리는 Member 엔티티 객체를 반환해.

장점: 엔티티 전체가 필요할 때 간단하게 쿼리할 수 있음.

단점: 불필요한 필드나 데이터까지 모두 메모리에 로드되므로 성능 상 이슈가 생길 수 있음.

 

2.2 필드 프로젝션 (스칼라 프로젝션)

 

필드 프로젝션은 필요한 필드만 선택하여 반환하는 방식으로, 엔티티의 전체 속성을 모두 가져오지 않고, 일부 속성만 가져옴.

 

예시:

SELECT m.name, m.age FROM Member m

 

이 쿼리는 Member 엔티티에서 nameage 필드만 반환해.

반환 값: Object[] 형태로 반환됨. 각 배열의 값은 순서대로 nameage에 매핑됨.

예: List<Object[]> result = query.getResultList();

Object[][name, age] 형태.

장점: 특정 필드만 가져오기 때문에 성능이 더 좋고, 데이터 트래픽을 줄일 수 있음.

단점: 반환 값이 엔티티가 아니기 때문에 타입 안전성이 떨어지며, 필드 순서에 따라 배열을 처리해야 함.

 

2.3 DTO 프로젝션

 

DTO(Data Transfer Object)를 사용하여 필요한 필드를 매핑하고, 지정된 객체 형태로 반환하는 방식이야. 이 방법은 타입 안전성을 제공하며, 필드 기반의 스칼라 프로젝션보다 사용하기 편리해.

 

예시:

SELECT new com.example.dto.MemberDTO(m.name, m.age) FROM Member m

 

MemberDTOm.namem.age 값을 받는 생성자를 가짐.

DTO 객체 생성: new 키워드를 사용하여 DTO를 생성하고 반환.

장점: 타입 안전성 보장, 가독성이 좋고 복잡한 비즈니스 로직을 DTO에 담을 수 있음.

단점: DTO의 생성자를 올바르게 만들어야 하며, 복잡한 쿼리일 경우 코드가 복잡해질 수 있음.

 

MemberDTO 예시:

public class MemberDTO {
    private String name;
    private int age;

    public MemberDTO(String name, int age) {
        this.name = name;
        this.age = age;
    }

    // Getters and Setters
}

 

3. 프로젝션의 사용 예시

 

3.1 단일 필드 프로젝션

SELECT m.name FROM Member m WHERE m.age > 30

 

이 쿼리는 30세 이상의 회원 이름만을 반환해.

반환 값은 List<String> 형태.

 

3.2 다중 필드 프로젝션

SELECT m.name, m.age FROM Member m WHERE m.age > 30

 

회원의 이름과 나이만 반환.

반환 값은 List<Object[]>로, 각 Object[]nameage로 구성됨.

 

3.3 DTO 프로젝션

SELECT new com.example.dto.MemberDTO(m.name, m.age) FROM Member m WHERE m.age > 30

 

이 쿼리는 30세 이상의 회원을 MemberDTO로 변환하여 반환해.

반환 값은 List<MemberDTO> 형태.

 

4. 필드 프로젝션과 DTO 프로젝션의 차이점

 

필드 프로젝션은 결과를 Object[] 형태로 반환하지만, DTO 프로젝션은 특정 클래스로 반환되므로 코드 가독성과 유지보수 측면에서 DTO 프로젝션이 더 나아.

필드 프로젝션은 간단한 경우에만 적합하며, 복잡한 데이터 구조를 반환해야 하는 경우에는 DTO 프로젝션을 사용하는 것이 좋음.

 

5. 실전에서의 활용

 

프로젝션은 데이터베이스와 상호작용할 때 효율적인 방법을 제공하며, 다음과 같은 상황에서 주로 사용돼:

 

성능 최적화: 불필요한 데이터를 가져오지 않고 필요한 필드만 조회할 때.

보고서 생성: 엔티티의 일부 필드만 필요한 경우, DTO로 간편하게 데이터를 매핑할 수 있음.

복잡한 비즈니스 로직 처리: DTO를 활용하여 복잡한 로직을 처리하거나 계산된 데이터를 반환할 때.

 

6. 주의 사항

 

필드 프로젝션의 타입 안전성: 필드 프로젝션은 Object[] 형태로 반환되므로, 타입 안전성이 떨어질 수 있어. 만약 타입 안정성을 원한다면 DTO 프로젝션을 사용하는 것이 좋음.

쿼리 복잡도: DTO 프로젝션을 사용하려면 DTO 클래스에 올바른 생성자가 필요하고, 쿼리가 복잡해질 수 있음. 따라서, DTO가 복잡해지면 쿼리의 복잡성도 고려해야 함.

 

결론

 

JPQL의 프로젝션은 데이터베이스에서 필요한 데이터만 선택적으로 조회하는 방법으로, 성능 최적화와 유지보수성을 높이는 데 중요한 역할을 해. 엔티티 프로젝션, 필드 프로젝션, DTO 프로젝션 세 가지 방법을 적절히 활용하여, 필요에 맞는 데이터를 효율적으로 조회하는 것이 핵심이야.

 

 


이 정보는 https://github.com/MoochiPark/jpa/tree/master/chapter10에서 자세하게 확인할 수 있습니다.

 

1. JPQL 기본 문법

 

JPQL은 객체를 대상으로 하는 쿼리 언어로 SQL과 유사하게 동작하지만, 테이블이 아닌 엔티티를 조회한다.

예를 들어, Member 엔티티에서 이름이 ‘John’인 회원을 조회하는 JPQL 쿼리는 다음과 같아.

String jpql = "SELECT m FROM Member m WHERE m.name = 'John'";
List<Member> members = em.createQuery(jpql, Member.class).getResultList();

 

2. 프로젝션 (Projection)

 

프로젝션은 SELECT 절에 조회할 대상을 지정하는 것을 의미해. 엔티티, 임베디드 타입, 스칼라 타입(기본 데이터 타입) 등이 프로젝션 대상이 될 수 있어.

 

엔티티 프로젝션:

String jpql = "SELECT m FROM Member m";
List<Member> members = em.createQuery(jpql, Member.class).getResultList();

 

이 경우, Member 엔티티 자체를 조회하며, 결과로 영속성 컨텍스트에서 관리되는 Member 엔티티 리스트가 반환돼.

 

스칼라 타입 프로젝션:

String jpql = "SELECT m.name, m.age FROM Member m";
List<Object[]> result = em.createQuery(jpql).getResultList();

 

여기서는 nameage 같은 기본 데이터 타입을 조회해. 결과는 Object 배열로 반환돼서, 인덱스로 접근해야 해.

 

DTO 프로젝션:

String jpql = "SELECT new com.example.dto.MemberDTO(m.name, m.age) FROM Member m";
List<MemberDTO> result = em.createQuery(jpql, MemberDTO.class).getResultList();

 

이 경우 JPQL에서 new 키워드를 사용해 특정 DTO 객체를 반환할 수 있어. DTO 클래스에 맞는 생성자가 필요하며, 패키지 이름까지 모두 명시해야 해.

 

3. 파라미터 바인딩

 

JPQL에서 쿼리를 실행할 때 동적으로 값을 입력할 수 있어. 이름 기준 파라미터 바인딩위치 기준 파라미터 바인딩이 사용될 수 있음.

 

이름 기준 파라미터 바인딩:

String jpql = "SELECT m FROM Member m WHERE m.name = :name";
List<Member> members = em.createQuery(jpql, Member.class)
                         .setParameter("name", "John")
                         .getResultList();

 

위치 기준 파라미터 바인딩:

String jpql = "SELECT m FROM Member m WHERE m.name = ?1";
List<Member> members = em.createQuery(jpql, Member.class)
                         .setParameter(1, "John")
                         .getResultList();

 

위치 기준보다 이름 기준이 더 명확하고 유지보수하기 쉬워서 주로 사용돼.

 

4. 페이징 처리

 

JPA는 데이터베이스 페이징 처리를 쉽게 할 수 있도록 **setFirstResult()**와 setMaxResults() 메서드를 제공해.

List<Member> members = em.createQuery("SELECT m FROM Member m ORDER BY m.name DESC", Member.class)
                         .setFirstResult(10) // 시작 위치
                         .setMaxResults(20)  // 조회할 데이터 수
                         .getResultList();

 

이 코드는 name을 기준으로 정렬된 회원 목록에서 11번째부터 30번째까지의 회원을 조회하는 예제야.

 

5. 조인

 

JPQL에서 조인을 사용해 엔티티 간의 관계를 탐색할 수 있어. 내부 조인, 외부 조인, 컬렉션 조인 등이 있음.

 

내부 조인:

String jpql = "SELECT m FROM Member m JOIN m.team t WHERE t.name = :teamName";
List<Member> members = em.createQuery(jpql, Member.class)
                         .setParameter("teamName", "TeamA")
                         .getResultList();

 

Member 엔티티에서 Team과 조인을 통해 팀 이름이 ‘TeamA’인 팀의 모든 회원을 조회하는 예제야.

 

페치 조인 (Fetch Join):

 

페치 조인성능 최적화를 위해 사용하는 조인으로, 연관된 엔티티를 한 번에 조회할 수 있게 해줘.

String jpql = "SELECT m FROM Member m JOIN FETCH m.team";
List<Member> members = em.createQuery(jpql, Member.class).getResultList();

 

이 코드는 회원을 조회하면서 연관된 팀도 함께 조회해 지연 로딩 문제를 해결해.

 

6. 집합 함수와 정렬

 

집합 함수는 데이터베이스의 통계 데이터를 조회할 때 사용돼.

String jpql = "SELECT COUNT(m), SUM(m.age), AVG(m.age) FROM Member m";
Object[] result = em.createQuery(jpql).getSingleResult();
System.out.println("회원 수: " + result[0]);
System.out.println("나이 합계: " + result[1]);
System.out.println("평균 나이: " + result[2]);

 

이 코드는 Member 엔티티의 수, 나이의 합, 평균 나이를 출력하는 예제야.

 

7. 서브 쿼리

 

JPQL에서도 서브 쿼리를 사용할 수 있어. 다만, JPQL은 WHEREHAVING 절에서만 서브 쿼리를 지원함.

String jpql = "SELECT m FROM Member m WHERE m.age > (SELECT AVG(m2.age) FROM Member m2)";
List<Member> members = em.createQuery(jpql, Member.class).getResultList();

 

이 코드는 평균 나이보다 많은 회원을 조회하는 예제야.

 

8. 벌크 연산

 

벌크 연산은 여러 엔티티를 한 번에 업데이트하거나 삭제할 때 사용돼. 벌크 연산 후에는 영속성 컨텍스트를 비우거나 다시 조회해야 함.

String jpql = "UPDATE Member m SET m.age = 20 WHERE m.age < 18";
int resultCount = em.createQuery(jpql).executeUpdate();
System.out.println(resultCount + "명의 회원의 나이를 업데이트했습니다.");

 

이 예시는 나이가 18세 미만인 모든 회원의 나이를 20세로 업데이트하는 벌크 연산 예제야.

  • 네이버 블로그 공유
  • 네이버 밴드 공유
  • 페이스북 공유
  • 카카오스토리 공유