728x90
- CQRS : 명령 모델과 조회 모델을 분리하는 패턴
- 명령 모델 : 상태를 변경하는 기능 구현 ex) 회원 가입, 주문 취소
- 조회 모델 : 데이터를 보여주는 기능 ex) 주문 목록, 주문 상세, 회원 상태
- 스프링 데이터 JPA 간단 예제
// Entity 클래스 생성
@Entity
@Getter
public class Book {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
private String title;
private String author;
}
// Repository 인터페이스 생성
public interface BookRepository extends JpaRepository<Book, Long> {
}
// Service 클래스 생성
@Service
public class BookService {
private final BookRepository bookRepository;
@Autowired
public BookService(BookRepository bookRepository) {
this.bookRepository = bookRepository;
}
public List<Book> findAllBooks() {
return bookRepository.findAll();
}
public Optional<Book> findBookById(Long id) {
return bookRepository.findById(id);
}
public Book saveBook(Book book) {
return bookRepository.save(book);
}
public void deleteBook(Long id) {
bookRepository.deleteById(id);
}
}
BookRepository 인터페이스에서 다양한 메서드를 작성할 수 있다.
public interface BookRepository extends JpaRepository<Book, Long> {
List<Book> findByAuthor(String author);
// JPQL
@Query("SELECT b FROM Book b WHERE b.title LIKE %:keyword%")
List<Book> findByTitleContaining(@Param("keyword") String keyword);
// Native Query
@Query(value = "SELECT * FROM books WHERE author = :author", nativeQuery = true)
List<Book> findByAuthorWithNativeQuery(@Param("author") String author);
// 페이징(Paging) 처리
// 제목에 특정 단어가 포함된 책을 검색하고 페이징 처리
@Query("SELECT b FROM Book b WHERE b.title LIKE %:keyword%")
Page<Book> findByTitleContaining(@Param("keyword") String keyword, Pageable pageable);
}
// Controller ( 페이징 처리 사용 )
@GetMapping("/books")
public Page<Book> getBooksByTitleContaining(
@RequestParam String keyword,
@RequestParam(defaultValue = "0") int page, // 0 (첫번째 페이지)
@RequestParam(defaultValue = "10") int size) { // 10개 가져온다.
return bookRepository.findByTitleContaining(keyword, PageRequest.of(page, size));
}
- 동적쿼리 → Queydsl 을 사용하자!
public PageImpl<CouponMasterAdminResDto> findAllCouponMasterByDateOrCouponName(
Pageable pageable, String searchStartDate, String searchEndDate, String selectSearchCondition, String searchCondition,
String selectStatus, String searchMainCategory) {
LocalDateTime startDateTime = DateTimeUtils.parseStartDateTime(searchStartDate);
LocalDateTime endDateTime = DateTimeUtils.parseEndDateTime(searchEndDate);
List<CouponMasterAdminResDto> list = queryFactory
.select(new QCouponMasterAdminResDto(
// 필요한 객체
))
.from(couponMaster)
.where(
couponMaster.createdAt.between(startDateTime, endDateTime),
searchContainsCouponName(selectSearchCondition, searchCondition), // 동적쿼리
searchIsUsableOrIsExposed(selectStatus),
searchFilterByMainCategory(searchMainCategory)
)
.orderBy(couponMaster.createdAt.desc())
.offset(pageable.getOffset())
.limit(pageable.getPageSize())
.fetch();
private BooleanExpression searchContainsCouponName(String selectSearchCondition, String searchCondition) {
switch (selectSearchCondition) {
case "couponName":
return couponMaster.couponName.contains(searchCondition);
case "couponMasterUuid":
return couponMaster.masterId.eq(searchCondition);
case "createUser":
return couponMaster.createdUser.contains(searchCondition);
case "modifiedUser":
return couponMaster.modifiedUser.contains(searchCondition);
}
return couponMaster.couponName.contains(searchCondition)
.or(couponMaster.masterId.eq(searchCondition))
.or(couponMaster.createdUser.contains(searchCondition))
.or(couponMaster.modifiedUser.contains(searchCondition));
}
728x90
'Books > 도메인 주도 개발 시작하기' 카테고리의 다른 글
Chapter7~8. 도메인서비스 ~ 애그리거트 트랜잭션 관리 (0) | 2024.04.10 |
---|---|
Chapter6. 응용 서비스와 표현 영역 (0) | 2024.04.08 |
Chapter4. 리포지터리와 모델 구현 (0) | 2024.04.08 |
Chapter3. 애그리거트 (0) | 2024.04.07 |
Chapter2. 아키텍처 개요 (0) | 2024.04.05 |
댓글