public interface MemberRepository extends JpaRepository<Member,Long> {
//실제 아무것도 없음. }
<S extends T> S save(S entity)
void delete(ID id)
Optional<T> findById(ID id)
Iterable<T> findAll()
long count()
기타 등등...
public interface MemberRepository extends Repository<Member, Long> {
List<User> findByEmailAndName(String email, String name); }
[생성된 JPQL]
select m from Member m where m.email = ?1
and m.name = ?2
[인터페이스에 쿼리작성 가능]
public interface UserRepository extends JpaRepository<User, Long> {
@Query("select u from User u where u.emailAddress = ?1")
User findByEmailAddress(String emailAddress); }
[JPA 네이티브 쿼리 지원]
public interface UserRepository extends JpaRepository<User, Long> {
@Query(value = "SELECT * FROM USERS WHERE EMAIL_ADDRESS = ?0", nativeQuery = true)
User findByEmailAddress(String emailAddress); }
[수정 쿼리도 직접 정의 가능]
@Modifying(clearAutomatically = true)
@Query("update User u set u.firstname = ?1 where u.lastname = ? 2")
int setFixedFirstnameFor(String firstname, String lastname);
가장 중요한 것은 JPA 자체를 이해하는 것!
JpaRepository
인터페이스를 통해서 기본적인 CRUD 기능 제공한다. CrudRepository
에서 fineOne()
->findById()
로 변경되었다.JpaRepository
인터페이스만 상속받으면 스프링 데이터 JPA가 프록시 기술을 사용해서 구현 클래스를 만들 어준다. 그리고 만든 구현 클래스의 인스턴스를 만들어서 스프링 빈으로 등록한다.스프링 데이터 JPA는 인터페이스에 메서드만 적어두면, 메서드 이름을 분석해서 쿼리를 자동으로 만들고 실행해주는 기능을 제공한다.
순수 JPA 리포지토리
public List<Member> findByUsernameAndAgeGreaterThan(String username, int age) {
return em.createQuery("select m from Member m where m.username = :username
and m.age > :age")
.setParameter("username", username)
.setParameter("age", age)
.getResultList();
}
순수 JPA를 사용하면 직접 JPQL을 작성하고, 파라미터도 직접 바인딩 해야 한다.
스프링 데이터 JPA
public interface MemberRepository extends JpaRepository<Member, Long> {
List<Member> findByUsernameAndAgeGreaterThan(String username, int age);
}
스프링 데이터 JPA가 제공하는 쿼리 메소드 기능
find...By
,read...By
, query...By
, get...By
findHelloBy
처럼 ...에 식별하기 위한 내용(설명)이 들어가도 된다.count...By
반환타입 long
exists...By
반환타입 boolean
delete...By
, remove...By
반환타입 long
findDistinct
, findMemberDistinctBy
findFirst3
, findFirst
, findTop
, findTop3
JPQL 직접 사용하기
public interface SpringDataJpaItemRepository extends JpaRepository<Item, Long> { //쿼리 메서드 기능
List<Item> findByItemNameLike(String itemName);
//쿼리 직접 실행
@Query("select i from Item i where i.itemName like :itemName and i.price
<= :price")
List<Item> findItems(@Param("itemName") String itemName, @Param("price")
Integer price);
}
쿼리 메서드 기능 대신에 직접 JPQL을 사용하고 싶을 때는 @Query
와 함께 JPQL을 작성하면 된다. 이때는 메 서드 이름으로 실행하는 규칙은 무시된다.
참고로 스프링 데이터 JPA는 JPQL 뿐만 아니라 JPA의 네이티브 쿼리 기능도 지원하는데, JPQL 대신에 SQL 을 직접 작성할 수 있다.
public interface SpringDataJpaItemRepository extends JpaRepository<Item, Long> {
List<Item> findByItemNameLike(String itemName);
List<Item> findByPriceLessThanEqual(Integer price);
//쿼리 메서드
// List<Item> findByItemNameLikeAndPriceLessThanEqual(String itemName, Integer price);
//쿼리 직접실행
@Query("select i from Item i where i.itemName like :itemName and i.price <= :price")
List<Item> findItems(@Param("itemName")String itemName, @Param("price") Integer price);
}
JpaRepository
인터페이스를 인터페이스 상속 받으면 기본적인 CRUD 기능 을 사용할 수 있다.@Query
를 사용해서 직접 쿼리를 실행하면 된다.동적 쿼리를 사용하면 좋겠지만, 스프링 데이터 JPA는 동적 쿼리에 약하다. 이번에는 직접 4가지 상황을 스프링 데이터 JPA로 구현해보자.
그리고 이 문제는 이후에 Querydsl에서 동적 쿼리로 깔끔하게 해결할 수 있다.
스프링 데이터 JPA도
Example
이라는 기능으로 약간의 동적 쿼리를 지원하지만, 실무에서 사용하기는 기능이 빈약하다. 실무에서 JPQL 동적 쿼리는 Querydsl을 사용하는 것이 좋다.
findAll()
코드에는 보이지 않지만 JpaRepository
공통 인터페이스가 제공하는 기능이다. 모든 Item
을 조회한다.
다음과 같은 JPQL이 실행된다.
select i from Item i
findByItemNameLike()
이름 조건만 검색했을 때 사용하는 쿼리 메서드이다. 다음과 같은 JPQL이 실행된다.
select i from Item i where i.name like ?
findByPriceLessThanEqual()
가격 조건만 검색했을 때 사용하는 쿼리 메서드이다. 다음과 같은 JPQL이 실행된다.
select i from Item i where i.price <= ?
findByItemNameLikeAndPriceLessThanEqual() 이름과 가격 조건을 검색했을 때 사용하는 쿼리 메서드이다. 다음과 같은 JPQL이 실행된다.
select i from Item i where i.itemName like ? and i.price <= ?
findItems()
메서드 이름으로 쿼리를 실행하는 기능은 다음과 같은 단점이 있다.
@Query
애노테이션을 사용하면 된다.@Param("itemName")
애노테이션을 사용하고, 애노테이션의 값에 파라미터 이름을 주면 된다.근데 JPARepository를 자동으로 생성해줬다고 해서 Service에서 바로 갖다 쓰면 OCP, DIP를 위반하게 된다.
그래서 기존에 ItemRepository를 구현해서 거기서 SpringDataJpaItemRepository를 주입 받아 사용하자.
@RequiredArgsConstructor
public class JpaItemRepositroyV2 implements ItemRepository {
private final SpringDataJpaItemRepository repository;
@Override
public Item save(Item item) {
return repository.save(item);
}
@Override
public void update(Long itemId, ItemUpdateDto updateParam) {
Item findItem = repository.findById(itemId).orElseThrow();
findItem.setItemName(updateParam.getItemName());
findItem.setPrice(updateParam.getPrice());
findItem.setQuantity(updateParam.getQuantity());
}
@Override
public Optional<Item> findById(Long id) {
return repository.findById(id);
}
@Override
public List<Item> findAll(ItemSearchCond cond) {
String itemName = cond.getItemName();
Integer maxPrice = cond.getMaxPrice();
if (StringUtils.hasText(itemName) && maxPrice != null) {
// return repository.findByItemNameLikeAndPriceLessThanEqual("%" + itemName + "%", maxPrice);
return repository.findItems("%" + itemName + "%", maxPrice);
} else if (StringUtils.hasText(itemName)) {
return repository.findByItemNameLike("%" + itemName + "%");
} else if (maxPrice != null) {
return repository.findByPriceLessThanEqual(maxPrice);
} else {
return repository.findAll();
}
}
}
JpaItemRepositoryV2
는 ItemRepository
를 구현한다. 그리고 SpringDataJpaItemRepository
를 사용한다.
런타임의 객체 의존관계는 다음과 같이 동작한다.
itemService
-> jpaItemRepositoryV2
-> springDataJpaItemRepository(프록시 객체)
이렇게 중간에서 JpaItemRepository
가 어댑터 역할을 해준 덕분에 ItemService
가 사용하는 ItemRepository
인터페이스를 그대로 유지할 수 있고 클라이언트인 ItemService
의 코드를 변경하지 않아도 되는 장점이 있다.
다음으로 기능에 대해서 알아보자.
save() repository.save(item)
스프링 데이터 JPA가 제공하는 save()
를 호출한다.
update()
스프링 데이터 JPA가 제공하는 findById()
메서드를 사용해서 엔티티를 찾는다.
그리고 데이터를 수정한다.
이후 트랜잭션이 커밋될 때 변경 내용이 데이터베이스에 반영된다. (JPA가 제공하는 기능이다.)
findById() repository.findById(itemId)
스프링 데이터 JPA가 제공하는 findById()
메서드를 사용해서 엔티티를 찾는다.
findAll()
데이터를 조건에 따라 4가지로 분류해서 검색한다.
모든 데이터 조회
이름 조회
가격 조회
이름 + 가격 조회
모든 조건에 부합할 때는 findByItemNameLikeAndPriceLessThanEqual()
를 사용해도 되고, repository.findItems()
를 사용해도 된다. 그런데 보는 것 처럼 조건이 2개만 되어도 이름이 너무 길어지는 단
점이 있다. 따라서 스프링 데이터 JPA가 제공하는 메서드 이름으로 쿼리를 자동으로 만들어주는 기능과 @Query
로 직 접 쿼리를 작성하는 기능 중에 적절한 선택이 필요하다.
추가로 코드를 잘 보면 동적 쿼리가 아니라 상황에 따라 각각 스프링 데이터 JPA의 메서드를 호출해서 상당히 비효율 적인 코드인 것을 알 수 있다. 앞서 이야기했듯이 스프링 데이터 JPA는 동적 쿼리 기능에 대한 지원이 매우 약하다. 이 부분은 이후에 Querydsl을 사용해서 개선해보자.
테스트를 실행하자
먼저 ItemRepositoryTest
를 통해서 리포지토리가 정상 동작하는지 확인해보자. 테스트가 모두 성공해야 한다.
애플리케이션을 실행하자
ItemServiceApplication
를 실행해서 애플리케이션이 정상 동작하는지 확인해보자.
예외 변환
스프링 데이터 JPA도 스프링 예외 추상화를 지원한다. 스프링 데이터 JPA가 만들어주는 프록시에서 이미 예외 변환을 처리하기 때문에, @Repository
와 관계없이 예외가 변환된다.
주의! - 하이버네이트 버그
하이버네이트5.6.6
~5.6.7
을 사용하면Like
문장을 사용할 때 다음 예외가 발생한다. 스프링 부트2.6.5
버전은 문제가 되는 하이버네이트 5.6.7을 사용한다.
java.lang.IllegalArgumentException: Parameter value [\] did not match expected type [java.lang.String (n/a)]
build.gradle에 다음을 추가해서 하이버네이트 버전을 문제가 없는5.6.5.Final
로 맞추자.ext["hibernate.version"] = "5.6.5.Final"
스프링 데이터 JPA의 대표적인 기능을 알아보았다. 스프링 데이터 JPA는 이 외에도 정말 수 많은 편리한 기능을 제공 한다. 심지어 우리가 어렵게 사용하는 페이징을 위한 기능들도 제공한다. 스프링 데이터 JPA는 단순히 편리함을 넘어서 많은 개발자들이 똑같은 코드로 중복 개발하는 부분을 개선해준다. 개인적(김영한)으로 스프링 데이터 JPA는 실무에서 기본으로 선택하는 기술이다.