-
Spring 기본2 - 주문 할인 도메인 설계FrameWork/Spring&Spring-boot 2023. 11. 7. 18:08
주문과 할인 도메인 설계
- 주문과 할인 정책
- 회원은 상품을 주문할 수 있다.
- 회원은 등급에 따라 할인 정책을 적용할 수 있다.
- 할인 정책은 모든 VIP는 1000원은 할인해주는 고정 금액 할인을 적용해달라.(나중에 변경 될 수 있다.)
- 할인 정책은 변경 가능성이 높다. 회사의 할인 정책은 아직 확정되지 않았고, 오픈 직전까지 고민을 미루고 싶다. 최악의 경우 할인을 적용하지 않을 수 도 있다.
주문 도메인 협력,역할,책임
1. 주문생성: 클라이언트는 주문 서비스에 주문 생성을 요청한다.
2. 회원 조회 : 할인을 위해서는 회원 등급이 필요하다. 그래서 주문 서비스는 회우너 저장소에서 회원을 조회한다.
3. 할인 적용 : 주문 서비스는 회원 등급에 따른 할인 여부를 할인 정책에 위임한다.
4. 주문 결과 반환 : 주문 서비스는 할인 결과를 포함한 주문 결과를 반환한다.
* 참고 : 실제로는 주문 데이터를 DB에 저장하겠지만, 예제가 너무 복잡해 질 수 있어서 생략하고, 단순히 주문 결과를 반환한다.
시나리오
주문 서비스 역할은 주문 생성에 역할을 한다. 우선1) 주문을 생성하기 위해 회원 저장소 역할 에 해당 주문을 한 회원은 등급을 조회한다. 그리고 2)조회한 등급을 바탕으로 할인 정책 역할에게 할인을 받을 수 있는 지 물어본다. 그러고 3)나면 할인 정책 역할은 할인을 받을 수 있는지 여부를 할인 정책 역할이 반환하고 4)주문 서비스 역할은 최종적으로 할인 여부가지 적용된 주문을 클라이언트에게 반환해준다.
주문 도메인 전체(구현)
역할과 구현을 분리 해서 자유롭게 구현 객체를 조립할 수 있도록 설계되었다. 덕분에 회원 저장소는 물론이고, 할인 정책도 유연하게 변경할 수 있다.
주문 도메인 클래스 다이어그램(정적)
주문 도메인 객체(인스턴스) 다이어그램1(동적)
회원을 메모리에서 조회하고, 정액 할인 정책(고정 금액)을 지원해도 주문 서비스를 변경하지 않아도 된다. 역할들의 협력관계를 그대로 재사용할 수 있다.
주문 도메인 객체(인스턴스) 다이어그램2(동적)
회원을 메모리가 아닌 실제 DB에서 조회하고, 정률 할인 정책(주문 금액에 따라 %할인)을 지원해도 주문 서비스를 변경하지 않아도 된다. 즉 협력관계를 그대로 재사용할 수 있다.
OrderServiceImpl구현 코드예
package com.hello.core.order; import com.hello.core.discount.DiscountPolicy; import com.hello.core.discount.FixDiscountPolicy; import com.hello.core.member.Member; import com.hello.core.member.MemberRepository; import com.hello.core.member.MemoryMemberRepository; public class OrderServiceImpl implements OrderService{ private final MemberRepository memberRepository = new MemoryMemberRepository(); private final DiscountPolicy discountPolicy = new FixDiscountPolicy(); @Override public Order createOrder(Long memberId, String itemName, int itemPrice) { //1. 회원 정보 조회 Member member = memberRepository.findById(memberId); //2.할인에 대한 책임,역할은 discountPolicy에 위임 int discountPrice = discountPolicy.discount(member, itemPrice); return new Order(memberId,itemName,itemPrice,discountPrice); } }
OrderServiceTest구현 코드예
package com.hello.core.order; import com.hello.core.member.Grade; import com.hello.core.member.Member; import com.hello.core.member.MemberService; import com.hello.core.member.MemberServiceImpl; import org.junit.jupiter.api.Assertions; import org.junit.jupiter.api.Test; public class OrderServiceTest { MemberService memberService = new MemberServiceImpl(); OrderService orderService = new OrderServiceImpl(); @Test void createOrder(){ //primative type은 null을 넣을 수 없다. Long memberId = 1L; Member member = new Member(memberId, "memberA", Grade.VIP); memberService.join(member); Order order = orderService.createOrder(memberId, "itemA", 10000); Assertions.assertEquals(order.getDiscountPrice(),1000); } }
[출처 - 스프링 핵심 원리 - 기본편]
'FrameWork > Spring&Spring-boot' 카테고리의 다른 글
Spring 기본 4 - 관심사의 분리 (1) 2023.11.09 Spring 기본3 - 객체 지향 원리 적용, OCP,DIP 위반의 문제 (1) 2023.11.09 Spring 기본1 - 회원 도메인 설계, SOLID 원칙 (0) 2023.11.07 @Configuration과 싱글톤 (0) 2022.01.11 싱글톤 방식의 주의점 (0) 2022.01.11 - 주문과 할인 정책