이번 섹션에서는 [데이터 접근 기술 - JPA]에 대해서 알아보자.
👉 목차는 다음과 같다.
1) JPA 시작
2) JPA 설정
3) JPA 적용1 - 개발
4) JPA 적용2 - 리포지토리 분석
5) JPA 적용3 - 예외 변환
바로 하나씩 알아보자.
스프링과 JPA는 자바 엔터프라이즈(기업) 시장의 주력 기술이다.
스프링이 DI 컨테이너를 포함한 애플리케이션 전반의 다양한 기능을 제공한다면, JPA는 ORM 데이터 접근 기술을 제공한다.
스프링 + 데이터 접근기술의 조합을 구글 트랜드로 비교했을 때
JPA는 스프링 만큼이나 방대하고, 학습해야할 분량도 많다. 하지만 한번 배워두면 데이터 접근 기술에서 매우 큰 생산성 향상을 얻을 수 있다. 대표적으로 JdbcTemplate이나 MyBatis 같은 SQL 매퍼 기술은 SQL을 개발자가 직접 작성해야 하지만, JPA를 사용하면 SQL도 JPA가 대신 작성하고 처리해준다.
실무에서는 JPA를 더욱 편리하게 사용하기 위해 스프링 데이터 JPA와 Querydsl이라는 기술을 함께 사용한다.
중요한 것은 JPA이다. 스프링 데이터 JPA, Querydsl은 JPA를 편리하게 사용하도록 도와주는 도구라 생각하면 된다.
이 강의에서는 모든 내용을 다루지는 않고, JPA와 스프링 데이터 JPA, 그리고 Querydsl로 이어지는 전체 그림을 볼 것이다. 그리고 이 기술들을 우리 애플리케이션에 적용하면서 자연스럽게 왜 사용해야 하는지, 그리고 어떤 장점이 있는지 이해할 수 있게 된다.
이렇게 전체 그림을 보고 나면 앞으로 어떻게 공부해야 할지 쉽게 접근할 수 있을 것이다.
spring-boot-starter-data-jpa
라이브러리를 사용하면 JPA와 스프링 데이터 JPA를 스프링 부트와 통합하고, 설정도 아주 간단히 할 수 있다.
👉 spring-boot-starter-data-jpa
라이브러리를 사용해서 간단히 설정하는 방법을 알아보자.
build.gradle
에 다음 의존 관계를 추가한다.spring-boot-starter-data-jpa
는 spring-boot-starter-jdbc
도 함께 포함(의존)한다. 따라서 해당 라이브러리 의존관계를 제거해도 된다. ( 참고로 mybatis-spring-boot-starter
도 spring-bootstarter-jdbc
를 포함하기 때문에 제거해도 된다. )jakarta.persistence-api
: JPA 인터페이스hibernate-core
: JPA 구현체인 하이버네이트 라이브러리spring-data-jpa
: 스프링 데이터 JPA 라이브러리application.properties
에 다음 설정을 추가하자. (로그와 관련된 설정)main - application.properties
test - application.properties
org.hibernate.SQL=DEBUG
: 하이버네이트가 생성하고 실행하는 SQL을 확인할 수 있다.org.hibernate.type.descriptor.sql.BasicBinder=TRACE
: SQL에 바인딩 되는 파라미터를 확인할 수 있다.spring.jpa.show-sql=true
: 참고로 이런 설정도 있다. 이전 설정은 logger
를 통해서 SQL이 출력된다. 이 설정은 System.out
콘솔을 통해서 SQL이 출력된다. 따라서 이 설정은 권장하지는 않는다. ( 둘다 켜면 logger
, System.out
둘다 로그가 출력되어서 같은 로그가 중복해서 출력된다. )JPA에서 가장 중요한 부분은 객체와 테이블을 매핑하는 것이다.
JPA가 제공하는 애노테이션을 사용해서 Item
객체와 테이블을 매핑해보자.
@Entity
: JPA가 사용하는 객체라는 뜻이다. 이 에노테이션이 있어야 JPA가 인식할 수 있다. 이렇게 @Entity
가 붙은 객체를 JPA에서는 엔티티라 한다.@Id
: 테이블의 PK와 해당 필드를 매핑한다.@GeneratedValue(strategy = GenerationType.IDENTITY)
: PK 생성 값을 데이터베이스에서 생성하는 IDENTITY
방식을 사용한다. 예) MySQL auto increment@Column
: 객체의 필드를 테이블의 컬럼과 매핑한다.name = "item_name"
: 객체는 itemName
이지만 테이블의 컬럼은 item_name
이므로 이렇게 매핑했다.length = 10
: JPA의 매핑 정보로 DDL( create table
)도 생성할 수 있는데, 그때 컬럼의 길이 값으로 활용된다. ( varchar 10
)@Column
을 생략할 경우, 필드의 이름을 테이블 컬럼 이름으로 사용한다. 참고로 지금처럼 스프링 부트와 통합해서 사용하면 필드 이름을 테이블 컬럼 명으로 변경할 때 객체 필드의 카멜 케이스를 테이블 컬럼의 언더스코어로 자동으로 변환해준다. ( itemName
-> item_name
, 따라서 위 예제의 @Column(name = "item_name")
를 생략해도 된다. )public
또는 protected
의 기본 생성자가 필수이다. 기본 생성자를 꼭 넣어주자. ( public Item() {}
)👉 이렇게 하면 기본 매핑은 모두 끝난다. 이제 JPA를 실제 사용하는 코드를 작성해보자.
private final EntityManager em
: 생성자를 보면 스프링을 통해 엔티티 매니저( EntityManager
) 라는 것을 주입받은 것을 확인할 수 있다. JPA의 모든 동작은 엔티티 매니저를 통해서 이루어진다. 엔티티 매니저는 내부에 데이터소스를 가지고 있고, 데이터베이스에 접근할 수 있다.@Transactional
: JPA의 모든 데이터 변경(등록, 수정, 삭제)은 트랜잭션 안에서 이루어져야 한다. 조회는 트랜잭션이 없어도 가능하다. 변경의 경우 일반적으로 서비스 계층에서 트랜잭션을 시작하기 때문에 문제가 없다. 하지만 이번 예제에서는 복잡한 비즈니스 로직이 없어서 서비스 계층에서 트랜잭션을 걸지 않았다. JPA에서는 데이터 변경시 트랜잭션이 필수다. 따라서 리포지토리에 트랜잭션을 걸어주었다. 다시한번 강조하지만 일반적으로는 비즈니스 로직을 시작하는 서비스 계층에 트랜잭션을 걸어주는 것이 맞다. (참고로 개발하다보면 repository에 걸때도 있다. service가 없거나 하는 등)JpaConfig
를 사용하도록 변경했다.ItemRepositoryTest
를 통해서 리포지토리가 정상 동작하는지 확인 )ItemServiceApplication
를 실행해서 애플리케이션이 정상 동작하는지 확인 )
✔️ 참고
EntityManagerFactory
, JPA 트랜잭션 매니저( JpaTransactionManager
), 데이터소스 등등 다양한 설정을 해야 한다. 스프링 부트는 이 과정을 모두 자동화 해준다. main()
메서드 부터 시작해서 JPA를 처음부터 어떻게 설정하는지는 JPA 기본편을 참고하자. 그리고 스프링 부트의 자동 설정은 JpaBaseConfiguration
를 참고하자.JpaItemRepositoryV1
코드를 분석해보자.
1) save() - 저장
em.persist(item)
: JPA에서 객체를 테이블에 저장할 때는 엔티티 매니저가 제공하는 persist()
메서드를 사용하면 된다.id
에 값이 빠져있는 것을 확인할 수 있다. PK 키 생성 전략을 IDENTITY
로 사용했기 때문에 JPA가 이런 쿼리를 만들어서 실행한 것이다. 물론 쿼리 실행 이후에 Item
객체의 id
필드에 데이터베이스가 생성한 PK값이 들어가게 된다. (JPA가 INSERT SQL 실행 이후에 생성된 ID 결과를 받아서 넣어준다.)2) update() - 수정
em.update()
같은 메서드를 전혀 호출하지 않았다. 그런데 어떻게 UPDATE SQL이 실행되는 것일까?@Commit
을 붙이면 확인할 수 있다. (또는 JPA가 제공하는 flush 라는 메서드를 사용해도 된다.)3) findById() - 단건 조회
find()
를 사용하고 조회 타입과, PK 값을 주면 된다. 그러면 JPA가 다음과 같은 조회 SQL을 만들어서 실행하고, 결과를 객체로 바로 변환해준다.4) findAll - 목록 조회
from
다음에 Item
엔티티 객체 이름이 들어간다. 엔티티 객체와 속성의 대소문자는 구분해야 한다.where price <= :maxPrice
query.setParameter("maxPrice", maxPrice)
동적 쿼리 문제
JPA의 경우 예외가 발생하면 JPA 예외가 발생하게 된다.
EntityManager
는 순수한 JPA 기술이고, 스프링과는 관계가 없다. 따라서 엔티티 매니저는 예외가 발생하면 JPA 관련 예외를 발생시킨다.PersistenceException
과 그 하위 예외를 발생시킨다.IllegalStateException
, IllegalArgumentException
을 발생시킬 수 있다.DataAccessException
)로 어떻게 변환할 수 있을까?@Repository
에 있다.
예외 변환 전
@Repository의 기능
@Repository
가 붙은 클래스는 컴포넌트 스캔의 대상이 된다.@Repository
가 붙은 클래스는 예외 변환 AOP의 적용 대상이 된다.PersistenceExceptionTranslator
)를 등록한다.예외 변환 후
@Repository
애노테이션만 있으면 스프링이 예외 변환을 처리하는 AOP를 만들어준다.
✔️ 참고
PersistenceExceptionTranslationPostProcessor
를 자동으로 등록하는데, 여기에서 @Repository
를 AOP 프록시로 만드는 어드바이저가 등록된다.EntityManagerFactoryUtils.convertJpaAccessExceptionIfPossible()
이다.강의를 듣고 정리한 글입니다. 코드와 그림 등의 출처는 김영한 강사님께 있습니다.