DEV ℧ Developer Diary

[JPA] JPQL - 경로 표현식

해당 포스트는 인프런 김영한님의 자바 ORM 표준 JPA 프로그래밍 - 기본편 을 듣고 정리한 글입니다.

경로 표현식

경로 표현식

  • (점)을 찍어 객체 그래프를 탐색하는 것
select m.username   -> 상태 필드
  from Member m
  join m.team t     -> 단일 값 연관 필드
  join m.orders o   -> 컬렉션 값 연관 필드
 where t.name = '팀A
  1. m.username : Member의 username 필드
  2. m.team : Member와 연관관계 가있는 단일 값 엔티티
  3. m.orders : Member에서 Collection값으로 가는 연관 필드

용어 정리

  • 상태 필드(state field): 단순히 값을 저장하기 위한 필드 (ex: m.username)
  • 연관 필드(association field): 연관관계를 위한 필드
    • 단일 값 연관 필드 : @ManyToOne, @OneToOne, 대상이 엔티티(ex: m.team)
    • 컬렉션 값 연관 필드 : @OneToMany, @ManyToMany, 대상이 List, Set 과 같은 컬렉션(ex: m.orders)

특징

  • 상태 필드(state field) : 경로 탐색의 끝, 탐색 X
String query = "SELECT m.username FROM Member m";
List<String> result = entityManager.createQuery(query, String.class)
                .getResultList();

m.username 에서 더이상 탐색을 들어갈 수가 없음

  • 단일 값 연관 경로 : 묵시적 내부 조인 (inner join) 발생, 탐색 O
String query2 = "SELECT m.team FROM Member m";
List<String> result2 = entityManager.createQuery(query2, String.class)
        .getResultList();

해당 쿼리를 실행하면, 쿼리의 의도와는 다르게 Member에서 team을 inner join을 해서 데이터를 가져오는것을 알 수 있다.

/* SELECT
        m.team
    FROM
        Member m */ select
            team1_.id as id1_3_,
            team1_.age as age2_3_,
            team1_.name as name3_3_
        from
            Member member0_
        inner join
            Team team1_
                on member0_.TEAM_ID=team1_.id

해당 쿼리는 실무에서 의도치 않은 데이터들을 가져와 부하를 가져 올 수 있기 때문에 주의해야 한다.

만약 해당 쿼리를 아래처럼 바꾸면 해당 쿼리도 상태 필드이다. team의 탐색의 끝이기 때문이다.

String query2 = "SELECT m.team.name FROM Member m";
List<String> result2 = entityManager.createQuery(query2, String.class)
        .getResultList();
  • 컬렉션 값 연관 경로 : 묵시적 내부 조인 발생, 탐색 X
    • FROM 절에서 명시적 조인을 통해 별칭을 얻으면 별칭을 통해 탐색 가능
String query3 = "SELECT t.members FROM Team t";
Collection result3 = entityManager.createQuery(query3, Collection.class)
        .getResultList();

해당 쿼리는 Member 엔티티가 아닌 List인 members 컬렉션을 가져오는 쿼리이다 보니 원하는 결과가 나오지 않을 수 있다.

/* SELECT
        t.members
    FROM
        Team t */ select
            members1_.id as id1_0_,
            members1_.age as age2_0_,
            members1_.TEAM_ID as team_id5_0_,
            members1_.type as type3_0_,
            members1_.username as username4_0_
        from
            Team team0_
        inner join
            Member members1_
                on team0_.id=members1_.TEAM_ID

위의 쿼리는 아래와 같이 내부 탐색을 할 수 없다.

  • 틀린 예시
SELECT t.members.username FROM Team t /* Exception 발생 */
  • 명시적으로 변환
SELECT m.username FROM Team t join t.members m /* members를 join해 m을 초기화 */

members 컬렉션을 join해서 해당 컬렉션의 엔티티를 가져올 수 있다.

상태 필드 경로 탐색

  • JPQL: select m.username, m.age from Member m
  • SQL: select m.username, m.age from Member m

단일 값 연관 경로 탐색

  • JPQL: select o.member from Order o
  • SQL: select m.* from Orders o inner join Member m on o.member_id = m.id

명시직 조인, 묵시적 조인

  • 명시적 조인: join 키워드 직접 사용
    • select m from Member m join m.team t
  • 묵시적 조인: 경로 표현식에 의해 묵시적으로 SQL 조인 발생(내부 조인만 가능)
    • select m.team from Member m

경로 표현식 - 예제

  • select o.member.team from Order o -> (성공) 하지만 join이 두번 이상 일어남
  • select t.members from Team -> (성공) 컬렉션으로 접근했기 때문에 실행 가능
  • select t.members.username from Team t -> (실패) 컬렉션 내부의 필드로 접근할 수 없다.
  • select m.username from Team t join t.members m -> (성공) 컬렉션을 명시적 join을 이용해 다시 선언한 뒤 접근했기 때문에 가능

주의사항

  • 항상 내부 조인
  • 컬렉션은 경로 탐색의 끝, 명시적 조인을 통해 별칭을 얻어야함
  • 경로 탐색은 주로 SELECT, WHERE 절에서 사용하지만 묵시적 조인으로 인해 SQL의 FROM (JOIN) 절에 영향을 줌

실무의 조언 (중요)

  • 가급적 묵시적 조인 대신에 명시적 조인 사용 갑자기 join이 걸려 원치않은 쿼리가 날라 갈 수 있음
  • 조인은 SQL 튜닝에 중요 포인트
  • 묵시적 조인은 조인이 일어나는 상황을 한눈에 파악하기 어려움