TDD and Refactoring Help

JPA Tips

Using JPA Specification

public interface MemeberRepository extends JpaRepository<Member, Long>, JpaSpecificationExecutor<Member> { interface Specs { static Specification<Member> byName(String name) { return (root, query, builder) -> builder.equal(root.get("name"), name); } static Specification<Member> byMemberLevel(MemberLevel level) { return (root, query, builder) -> builder.equal(root.get("level"), level); } static Specification<Member> orderByName(Specification<Member> spec) { return (root, query, builder) -> { query.orderBy(builder.asc(root.get("name"))); return spec.toPredicate(root, query, builder); }; } } }
  • Usage

memeberRepository.findAll( orderByName( byName("백명석") .and(byMemberLevel(MemberLevel.GOLD)) ) );

Paging Query

  • Repository

    public interface ShoppingCartRepository extends JpaRepository<ShoppingCart, Long> { @Query( value = """ select s from ShoppingCart s inner join s.cartItems cartItems where s.member.id = :memberId and cartItems.product.id = :productId""", countQuery = """ select count(s) from ShoppingCart s inner join s.cartItems cartItems where s.member.id = :memberId and cartItems.product.id = :productId""" ) Page<ShoppingCart> memberAndProduct(Long memberId, Long productId, Pageable pageable); }
  • Client

    Pageable pageable = PageRequest.of(0, 10, Sort.by("id")); shoppingCartRepository.memberAndProduct(memberId, productId, pageable);

Custom JPA Repository

  • JPA를 사용할 때 interface만 있는 jpa repository와 impl이 있는 커스텀 리파지토리를 하나의 인터페이를 통해 사용하는 방법

public interface CustomShoppingCartRepository { ShoppingCart findByMemberId(Long memberId); } public interface ShoppingCartRepository extends JpaRepository<ShoppingCart, Long> , CustomShoppingCartRepository { } public class CustomShoppingCartRepositoryImpl implements CustomShoppingCartRepository { private final EntityManager entityManager; public CustomShoppingCartRepositoryImpl(EntityManager entityManager) { this.entityManager = entityManager; } @Override public ShoppingCart findByMemberId(Long memberId) { return entityManager.createQuery( "select new pe.msbaek.tddcases.cart.domain. sc from ShoppingCart sc " + "join fetch sc.member m " + "join fetch sc.cartItems ci " + "join fetch ci.product p " + "where m.id = :memberId", ShoppingCart.class) .setParameter("memberId", memberId) .getSingleResult(); } }
  • 위와 같이 Custom Repository를 정의(CustomShoppingCartRepository)/구현(CustomShoppingCartRepositoryImpl)하고, EntityRepository(ShoppingCartRepository)를 정의하고

  • 클라이언트에서는 EntityRepository를 주입받아 사용하면 됨

  • 주의사항

    • CustomerEntityRepository의 구현체는 Impl을 붙여야 함(spring-boot의 관행). CustomerEntityRepositoryImpl

  • 이렇게 하면 구현은 별도의 클래스로 하지만...

    • 사용하는 곳에서는 하나의 인터페이스에만 의존성을 갖으면 될 것 같아요.

    • 그럼 모든 엔터티마다 repository를 만들지 않고, aggregate root 역할을 하는 녀석에게만 repository를 만들 수 있지 않을까 합니다.

Last modified: 14 January 2025