본문 바로가기

Spring/JPA & Hibernate

JPA 쿼리

반응형

가능한 선택지

  • JPQL
  • JPA Criteria
  • QueryDSL
  • Native SQL
  • JDBC API, MyBatis, SpringJdbcTemplate 함께 사용 

JPQL 이란?

JPA를 사용하면 엔티티 객체를 중심으로 개발이 가능하다.

검색을 할 때도  테이블이 아닌 엔티티 객체를 대상으로 검색한다.

하지만 모든 데이터베이스의 데이터를 객체로 변환해서 검색하는 것은 불가능하다.

애플리케이션이 필요한 데이터만 데이터베이스에서 불러오려면 결국 검색 조건이 포함된 SQL이 필요하다.

 

이러한 문제를 해결하기 위해

JPA는 SQL을 추상화한 JPQL이라는 객체지향 쿼리 언어를 제공한다.

SQL과 문법이 유사하고, SELECT, FROM, WHERE, GROUP BY, HAVING, JOIN 이 지원된다.

JPQL은 엔티티 객체를 대상으로 쿼리를 하는 것이고, SQL은 데이터베이스 테이블을 대상으로 검색한다.

SQL을 추상화해서 특정 데이테베이스 SQL에 의존하지 않는다.

이러한 특성으로 JPQL은 객체지향 쿼리라고 한다.

 

JPQL 예시


List<Book> bookList = entityManager.createQuery("SELECT b FROM Book b WHERE b.name = '%JPA%'", Book.class).getResultList();

맵핑 정보를 이용해  실제 데이터베이스에  실행한 쿼리

하지만, 비즈니스 로직 구현 중 쿼리를 동적으로 생성해야할 경우가 많다. 이런 경우 쿼리를 스트링으로 만들기 힘들다. 이러한 문제를 해결하기 위해 Criteria 를 사용할 수 있다.

Criteria


CriteriaBuilder cb = entityManager.getCriteriaBuilder();
CriteriaQuery<Book> query = cb.createQuery(Book.class);

Root<Book> bookRoot = query.from(Book.class);
query.select(bookRoot).where(cb.equal(bookRoot.get("name"), "JPA"));
List<Book> bookList = entityManager.createQuery(query).getResultList();

맵핑 정보를 이용해  실제 데이터베이스에  실행한 쿼리

Criteria Predicates를 이용한 Or 연산


CriteriaBuilder criteriaBuilder = entityManager.getCriteriaBuilder();
CriteriaQuery<Book> query = criteriaBuilder.createQuery(Book.class);
Root<Book> bookRoot = query.from(Book.class);

Predicate finalPredicate;
Predicate firstPredicate = criteriaBuilder.equal(bookRoot.get("name"), "%JAVA%");

if (user.getName().equals("jun")) {
  	finalPredicate = criteriaBuilder.or(firstPredicate, criteriaBuilder.equal(bookRoot.get("name"), "%JPA%"));
  } else {
  	finalPredicate = firstPredicate;
}

query.select(bookRoot).where(finalPredicate);
List<Book> bookList = entityManager.createQuery(query).getResultList();

문자가 아닌 자바 코드로 JPQL을 작성할 수 있다.

  • JPQL 빌더 역활
  • JPQ 공식 기능
  • 하지만 복잡하고 실용성이 없다.
  • Criteria 대신에 QueryDSL 사용을 권장한다.

네이티브 SQL 소개


String sql = "SELECT id, name FROM User WHERE name = 'jun'";
List<User> resultList = entityManager.createNativeQuery(sql, User.class).getResultList();

EntityManager를 이용한 쿼리는 SQL 을 실행하기 전에 내부적으로 플러시한다.


User jun = new User(1L, "jun");
entityManager.persist(jun);
String sql = "SELECT id, name FROM User WHERE name = 'jun'";
List<User> resultList = entityManager.createNativeQuery(sql, User.class).getResultList();
System.out.println("[BEFORE FLUSH] resultList size = " + resultList.size());
entityManager.flush();

네이티브 쿼리의 경우, 쿼리 실행 전에 내부적으로 플러시

JDBC 직접 사용, SpringJdbcTemplate

JPA를 사용하면서 JDBC 커넥션을 직접 사용하거나, 스프링 JdbcTemplate, 마이바티스등을 함께 사용 가능하다.

단 영속성 컨텍스트를 적절한 시점에 강제로 플러시해줘야 트랜잭션 내에서의 변경사항을 반영하여 검색할 수 있다.

 

반응형