[JPA] JPQL 개념, 기본 기능
JPQL ( Java Persistence Query Language )
- 엔티티 객체를 조회하는 객체지향 쿼리
- JPA Query를 작성하는 방법을 제공한다.
- Table column명 사용 대신, 엔티티이름, 속성을 사용함.
- ANSI 표준 SQL이 제공하는 기능을 유사하게 지원한다
- 엔티티 객체를 대상으로 쿼리한다.
- 엔티티 직접조회, 묵시적 조인, 다형성지원으로 SQL보다 간결하다.
JPQL 맛보기
DB
username | age | id | team |
jieun | 24 | jieun11 | a |
jin | 22 | jin12 | a |
kim | 25 | kin123 | b |
@Column : JPA에서 DB Table의 Column을 Mapping 할 때 @Column Annotation을 사용함
@Entity(name="Member") // name속성의 기본값 : 클래스명
public class Member{
@Column(name="name")
private String username;
//...
}
String jqpl = "select m from Member as m where m.username = 'kim'";
List<Member> resultList = em.createQuery(jpql, Member.class).getResultList();
Member = 엔티티 이름,
m.username == 엔티티 객체의 필드 이름
회원 이름이 'kim'인 엔티티를 조회하라.
createQuery(실행할 Jpql, 반환할 엔티티의 클래스 타입)
getResultList() : JPA가 JPQL을 SQL로 변환해 DB 조회 - > 조회한 결과로 엔티티를 생성해 반환
실행 JPQL
select m from Member as m where m.username = 'kim'
실제로 실행된 SQL
select
member.id as id,
member.age as age,
member.team_id as team,
member.name as name
from
Member member
where
member.name='kim'
JPQL 기본 문법
- SELECT, UPDATE, DELETE 가능
- INSERT 불가능 (EntityManager.persist()를 사용해 엔티티를 저장한다. )
select 별칭 from 엔티티명 별칭
select r form Review r
select m from Member as m
⏺ SELECT
select m from Member as m where m.username='jieun
- 엔티티와 속성은 대소문자 구분 O
Member != member
UserName != username
- JPQL키워드는 대소문자 구분 x (sql 쿼리도 그렇지)
SELECT == select
FROM == from
- JPQL은 별칭이 필수다
select username from Member m // X
select m.username from Member m // O
- AS는 생략 가능하다
Member AS m == Member m
⏺ 쿼리의 종류 : TypeQuery와 Query
- 작성한 JPQL을 실행하기 위한 쿼리 객체
TypeQuery : 반환 타입 명확한 지정 가능
TypedQuery<Member> query
= em.createQuery("SELECT m FROM Member m", Member.class);
List<Member> resultList = query.getResultList();
for (Member member : resultList) {
System.out.println("member = " + member);
}
Query : 반환 타입 명확한 지정 불가능
Query query
= em.createQuery("SELECT m.username, m.age FROM Member m");
List<Member> resultList = query.getResultList();
for (Object o : resultList) {
Object[] result = (Object[]) o; //결과가 둘 이상이면 Object[] 반환
System.out.println("username = " + result[0]);
System.out.println("age = " + result[1]);
}
em.createQuery()
- 두번째 파라미터에 반환타입 지정시 TypeQuery 반환, 지정 안할 경우 Query 반환
- 반환형이 불확실한 경우에 Query를 사용한다.
⏺ 결과 조회하기
JPQL에서 쿼리를 통한 결과를 받을 때 사용하는 메서드🔽🔽
query.getResultList() : 결과를 리스트로 반환
query.getSingleResult : 결과가 한개일 때 사용 (1개가 아니면 예외 발생 )
- 결과 존재 x : javax.persistence.NoResultException
- 결과가 둘 이상임 : javax.persistence.NonUniqueResultException
기본 JPQL 사용방법
⏺ 쿼리 생성
TypedQuery<T> EntityManager#createQuery(String ql, Class<T> resultClass)
(출처 유튜브 최범균님)
⏺ 검색 조건 지정
where + and, or, 괄호 등
select r from Review r where r.hotelId = :hotelId
select r from Review r where r.hotelId = ?
select r from Review r where r.hotelId = :hotelId and r.mark > :minMark
selet p from Player p where p.position = :pos or p.team.if = :teamId
⏺ 파라미터 바인딩
- 쿼리에 작성되는 특정 속성을 매개변수로 매핑 하는 것.
- 쿼리에 매개변수를 매핑하는 방식엔 이름기준, 위치기준 두가지가 존재.
이름 기준 파라미터 (Named Parameters)
- 파라미터를 이름으로 구분함.
- =: 연산자 를 사용한다 (파라미터 앞에 : 을 붙힌다)
query.setParameter("hotelId", "H-001")
Query query = em.createQuery("select m from Member m where m.username =: username")
.setParameter("username", usernameParam);
인덱스 기반 파라미터 (Positional Parameters)
- 0부터 시작
- =? 연산자 사용 (파라미터 앞에 ? 을 붙힌다)
query.setParameter(0,"H-001")
Query query = em.createQuery("select m from Member m where m.username =? 1")
.setParameter(1, usernameParam);
** 매개변수 바인딩시 이름 기반 바인딩 사용을 권장함 (나중에 새로운 매개변수 추가시 위치/인덱스 기반 바인딩은 순서가 밀려서)
- 숫자를 이용하면 어떤 위치의 매개변수가 무엇을 의미하는지 쉽게 파악 불가능
- 가독성이 떨어져 유지보수 할 때 비효율적
⏺ 정렬 (SQL과 동일)
order by 기준1 (,기준2..)
asc, desc 사용가능,
select r from Review r order by r.id
select r from Review r order by r.id asc
select r from Review r order by r.id desc
select p from Player p order by p.position, p.name
select p from Player p order by p.team.id, p.name
- default : 오름차순 정렬
⏺ 비교연산자
연산자 | example |
= | u.name = 'JPA' |
< > | o.state < > ? |
> >= < <= | p.salary > 2000 |
between | mc.expiryDate between ? and ? |
in, not in | o.mark in (1,2,3) |
like, not like | u.name like '%유%' |
is null, is not null | u.name is null |
출처
https://devraphy.tistory.com/566?category=1059635
https://www.youtube.com/watch?v=UtEhC68GTH0&list=PLwouWTPuIjUi9Sih9mEci4Rqhz1VqiQXX&index=18