스프링부트 너 뭐 돼?🤷‍♀️(11) - JPQL

joyfulwave·2022년 12월 9일
0

피할 수 없다면 즐기자! 스프링부트 너.. 뭐 돼?




📚 객체지향 쿼리 언어(JPQL)

  • JPA는 다양한 쿼리 방법을 지원
  • JPQL : 권장, 실무사용, 거의 모든 문제 해결 가능
  • JPA Criteria : 비권장, 자바코드를 짜서 JPA를 빌드해주는 generator 모음
  • QueryDSL : 권장, 실무사용
  • 네이티브 SQL : connect by...
  • JDBC API : MyBatis, SpringJdbcTemplate

📚 JPQL

  • JPA를 사용하면 엔티티 객체를 중심으로 개발
  • 문제는 검색 쿼리
  • 검색을 할 때도 테이블이 아닌 엔티티 객체를 대상으로 검색
  • JPA는 SQL을 추상화한 JPQL이라는 객체 지향 쿼리 언어 제공
  • SQL과 문법 유사, SELECT, FROM, WHERE, GROUP BY, ... 지원.
  • JPQL은 엔티티 객체를 대상으로 쿼리
  • SQL은 데이터베이스 테이블을 대상으로 쿼리
  • Java Persistence Query Language
  • SQL을 추상화해서 특정 데이터베이스에 의존하지 않음
  • JPA는 JPQL을 분석 후, 적절한 SQL을 만들어 데이터베이스를 조회

📌 규칙

⚫ 엔티티이름

  • 테이블명 대신 엔티티명을 사용, @Entity(name="")으로 설정 가능
  • 지정하지 않을 시 클래스명을 기본값으로 사용(추천)

⚫ 별칭은 필수

  • JPQL은 별칭을 필수
  • AS는 생략 가능

📌 JPQL문법

  • select
    select절 항목
    from절
    where절
    group by절
    having절
    order by절

  • update
    update절
    where절

  • delete
    delete절
    where절

📌 함수

  • count(m) // 회원수
  • sum(m.age) // 나이의 합
  • avg(m.age) // 평균 나이
  • max(m.age) // 최대 나이
  • min(m.age) // 최소 나이

📌 TypedQuery, Query

⚫ TypedQuery : 반환 타입이 명확할 때 사용

TypedQuery<Member> query = 
	("select m from member m", Member.class)

TypedQuery<String> query = 
	("select m.username from member m", String.class)

⚫ Query : 반환 타입이 명확하지 않을 때 사용

Query query = 
	("select m.username, m.age from member m", Member.class)

📌 결과 조회 API

⚫ query.getResultList()

  • 결과가 없으면 빈 리스트 반환
  • 빈 collection이 반환되기 때문에 nullPointerException에 대한 걱정은 안해도돼요.

⚫ query.getSinglerResult()

  • 결과가 정확히 하나 이거나, 단일 객체 반환할 때만 사용
  • 결과가 없을 경우 NoResultException 발생
  • 결과가 둘 이상이여도 NonUniqueResultException 발생

📌 파라미터 바인딩 - 이름 기준, 위치

⚫ 이름기준

package com.koreait.jpaitem;

import java.time.LocalDateTime;
import java.util.List;

import javax.persistence.EntityManager;
import javax.persistence.EntityManagerFactory;
import javax.persistence.EntityTransaction;
import javax.persistence.Persistence;
import javax.persistence.Query;
import javax.persistence.TypedQuery;

import com.koreait.jpaitem.embedded.Address;
import com.koreait.jpaitem.embedded.Member;
import com.koreait.jpaitem.embedded.Period;



public class JpaMain8 {

	public static void main(String[] args) {

		EntityManagerFactory emf = Persistence.createEntityManagerFactory("hello");
		EntityManager em = emf.createEntityManager();
		EntityTransaction tx = em.getTransaction();
		tx.begin();
		
		try {
			
			Member member = new Member();
			member.setUsername("member1");
			member.setAge(20);
			em.persist(member);
			
			// 파라미터 바인딩 - 이름 기준
			TypedQuery<Member> query = em.createQuery("select m from Member m where m.username = :username", Member.class);
			query.setParameter("username", "member1");
			Member result = query.getSingleResult();
			System.out.println("====================================\n\n\n");
			System.out.println("result : " + result.getUsername());
			System.out.println("\n\n\n====================================");
			
			tx.commit();			
			
		} catch (Exception e) {
			tx.rollback();			
		}finally {
			em.close();
			emf.close();
		}
		
	}

}

⚫ 위치기준(권장하지 않음)

package com.koreait.jpaitem;

import java.time.LocalDateTime;
import java.util.List;

import javax.persistence.EntityManager;
import javax.persistence.EntityManagerFactory;
import javax.persistence.EntityTransaction;
import javax.persistence.Persistence;
import javax.persistence.Query;
import javax.persistence.TypedQuery;

import com.koreait.jpaitem.embedded.Address;
import com.koreait.jpaitem.embedded.Member;
import com.koreait.jpaitem.embedded.Period;



public class JpaMain8 {

	public static void main(String[] args) {

		EntityManagerFactory emf = Persistence.createEntityManagerFactory("hello");
		EntityManager em = emf.createEntityManager();
		EntityTransaction tx = em.getTransaction();
		tx.begin();
		
		try {
			
			Member member = new Member();
			member.setUsername("member1");
			member.setAge(20);
			em.persist(member);
			
						
			// 파라미터 바인딩 - 위치 기준
			Member result = em.createQuery("select m from Member m where m.username = ?1", Member.class)
					.setParameter(1, "member1")
					.getSingleResult();
			System.out.println("====================================\n\n\n");
			System.out.println("result : " + result.getUsername());
			System.out.println("\n\n\n====================================");
			
			tx.commit();			
			
		} catch (Exception e) {
			tx.rollback();			
		}finally {
			em.close();
			emf.close();
		}
		
	}

}




무사히 적응할 그 날을 기대 ✔️




출처
https://media.giphy.com/media/kyUIknbbDNvID5XzU4/giphy.gif
https://media.giphy.com/media/A6aHBCFqlE0Rq/giphy.gif

0개의 댓글