[JPA] JPQL(객체지향쿼리언어) 소개

동동·2022년 4월 24일

JPA

목록 보기
15/18
post-thumbnail

JPA는 다양한 쿼리 방법을 지원
JPQL / JPA Criteria / QueryDSL / native SQL / JDBC API 직접 사용, MyBatis, SpringJdbcTemplate 함께 사용

JPQL

List<Member> result = em.createQuery(
	"select m from Member m where m.age > 18",
                    Member.class
            ).getResultList();

  • JPA를 사용하면 엔티티 객체를 중심으로 개발
  • 문제는 검색 쿼리
    검색을 할 때도 테이블이 아닌 엔티티 객체를 대상으로 검색
  • 모든 DB 데이터를 객체로 변환해서 검색하는 것은 불가능
  • 애플리케이션이 필요한 데이터만 DB에서 불러오려면 결국 검색 조건이 포함된 SQL이 필요
  • JPA는 SQL을 추상화한 JPQL이라는 객체 지향 쿼리 언어 제공
  • SQL과 문법 유사
    SELECT / FROM / WHERE / CROUP BY / HAVING / JOIN 지원
  • JPQL은 엔티티 객체 / SQL은 데이터 베이스 테이블을 대상으로 쿼리
  • 테이블이 아닌 객체를 대상으로 검색하는 객체 지향 쿼리
  • SQL을 추상화해서 특정 데이터베이스 SQL에 의존하지 않음
  • JPQL을 한마디로 정의하면 객체 지향 SQL
  • 동적 쿼리를 다루기 힘듦(보완해줄 빌더가 필요함?)

📌 Criteria 소개

CriteriaBuilder cb = em.getCriteriaBuilder();
CriteriaQuery<Member> query = cb.createQuery(Member.class);

Root<Member> m = query.from(Member.class);

CriteriaQuery<Member> cq = query.select(m).where(cb.equal(m.get("username"), "kim"));
em.createQuery(cq).getResultList();

  • 문자가 아닌 자바코드로 JPQL을 작성할 수 있음
  • JPQL 빌더 역할
  • JPA에서 공식으로 지원하는 기능
  • 장점은 동적 쿼리를 다루기 쉽고 자바코드이기 때문에 문법적 오류가 날 확률이 적음
  • 단점은 너무 복잡하고 실용성이 없음
  • 따라서 Criteria 대신에 QueryDSL 사용을 권장

📌 QueryDSL 소개

//JPQL : select m from Member m where m.age > 18
JPAFactoryQuery query = new JPAQueryFactory(em);
QMember m = QMember.member;

List<Member> list =
    query.selectFrom(m)
         .where(m.age.gt(10))
         .orderBy(m.name.desc())
         .fetch();
  • 문자가 아닌 자바코드로 JPQL을 작성할 수 있음
  • JPQL빌더 역할
  • 자바코드이기 때문에 컴파일 시점에 문법 오류를 찾을 수 있음
  • 동적쿼리 작성 편함
  • 단순하고 쉬움
  • 따라서 실무에서 사용을 권장

📌 Native SQL 소개

em.createNativeQuery("select MEMBER_ID, city, street, zipcode, USERNAME from MEMBER")
        .getResultList();

  • JPA가 제공하는 SQL을 직접 사용하는 기능
  • JPQL로 해결할 수 없는 특정 데이터베이스에 의존적인 기능
    예) 오라클 CONNECT BY, 특정 DB만 사용하는 SQL 힌트

📌 JDBC 직접 사용, SpringJdbcTemplate 등

  • JPA를 사용하면서 JDBC 커넥션을 직접 사용하거나, 스프링 JdbcTemplate, Mybatis 등을 함께 사용 가능
  • 단, 영속성 컨텍스트를 적절한 시점에 강제로 플러시 필요
    예) JPA를 우회해서 SQL을 실행하기 직전에 영속성 컨텍스트 수동 플러시
profile
괴발개발

0개의 댓글