섹션 2. 스프링 핵심 원리 이해1 - 예제 만들기(주문 도메인)

dev-mage·2023년 1월 10일
0
post-thumbnail

스프링 핵심 원리 - 기본편 - 인프런 | 강의

주문과 할인 도메인

주문과 할인 도메인 설계

  • 주문 도메인 협력, 역할, 책임

  1. 주문 생성: 클라이언트는 주문 서비스에 주문 생성을 요청
  2. 회원 조회: 할인을 위해서는 회원등급이 필요함 → 주문 서비스는 회원 저장소에서 회원을 조회
  3. 할인 적용: 주문 서비스는 회원 등급에 따른 할인 여부를 할인 정책에 위임
  4. 주문 결과 반환: 주문 서비스는 할인 결과를 포함한 주문 결과를 반환
    • 실제로는 상품 도메인도 있을 수 있고 주문 데이터를 데이터베이스에 저장도 하겠지만 여기서는 생략
  • 주문 도메인 전체(역할과 구현)
    • 역할과 구현을 분리해서 자유롭게 구현 객체를 조립할 수 있게 설계됨
    • 덕분에 회원 저장소는 물론이고 할인 정책도 유연하게 변경할 수 있음
    • 단일 책임 원칙을 잘 따른 설계
      • 만약 주문 서비스에 할인 관련 로직을 넣었다면 유연하게 변경할 수 없었을 것

  • 주문 도메인 클래스 다이어그램

  • 주문 도메인 객체 다이어그램1
    • 회원을 메모리에서 조회하고 정액 할인 정책(고정 금액)을 지원해도 주문 서비스를 변경하지 않아도 됨
    • 역할들의 협력 관계를 그대로 사용할 수 있음

  • 주문 도메인 객체 다이어그램2
    • 회원을 메모리가 아닌 실제 데이터베이스에서 조회하고 정률 할인 정책(주문 금액에 따라 % 할인)을 지원해도 주문 서비스를 변경하지 않아도 됨
    • 협력 관계를 그대로 재사용할 수 있음

주문과 할인 도메인 개발

  • 할인 정책 인터페이스
package hello.core.discount;

import hello.core.member.Member;

public interface DiscountPolicy {

    /**
     * @return 할인 금액
    * */
    int discount(Member member, int price);
}
  • 정액 할인 정책 구현체
package hello.core.discount;

import hello.core.member.Grade;
import hello.core.member.Member;

public class FixDiscountPolicy implements DiscountPolicy {
    private int discountFixAmount = 1000; // 1000원 할인

    @Override
    public int discount(Member member, int price) {
        if (member.getGrade() == Grade.VIP) {
            return discountFixAmount;
        } else {
            return 0;
        }
    }
}
  • 주문 엔티티
package hello.core.order;

public class Order {
    private Long memberId;
    private String itemName;
    private int itemPrice;
    private int discountPrice;

    public Order(Long memberId, String itemName, int itemPrice, int discountPrice) {
        this.memberId = memberId;
        this.itemName = itemName;
        this.itemPrice = itemPrice;
        this.discountPrice = discountPrice;
    }

    public int calculatePrice() {
        return itemPrice - discountPrice;
    }

    public Long getMemberId() {
        return memberId;
    }

    public void setMemberId(Long memberId) {
        this.memberId = memberId;
    }

    public String getItemName() {
        return itemName;
    }

    public void setItemName(String itemName) {
        this.itemName = itemName;
    }

    public int getItemPrice() {
        return itemPrice;
    }

    public void setItemPrice(int itemPrice) {
        this.itemPrice = itemPrice;
    }

    public int getDiscountPrice() {
        return discountPrice;
    }

    public void setDiscountPrice(int discountPrice) {
        this.discountPrice = discountPrice;
    }

    @Override
    public String toString() {
        return "Order{" +
                "memberId=" + memberId +
                ", itemName='" + itemName + '\'' +
                ", itemPrice=" + itemPrice +
                ", discountPrice=" + discountPrice +
                '}';
    }
}
  • 주문 서비스 인터페이스
package hello.core.order;

public interface OrderService {
    Order createOrder(Long memberId, String itemName, int itemPrice);
}
  • 주문 서비스 구현체
package hello.core.order;

import hello.core.discount.DiscountPolicy;
import hello.core.discount.FixDiscountPolicy;
import hello.core.member.Member;
import hello.core.member.MemberRepository;
import 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) {
        Member member = memberRepository.findById(memberId);
        // 단일 책임 원칙을 잘 지켰기 때문에 할인에 문제가 생기면 할인만 수정하면 됨
        int discountPrice = discountPolicy.discount(member, itemPrice);
        return new Order(memberId, itemName, itemPrice, discountPrice);
    }
}
  • 주문 생성 요청이 오면
    1. 회원 정보 조회
    2. 할인 정책 적용
    3. 주문 객체 생성 후 반환
    • 메모리 회원 저장소와 고정 금액 할인 정책을 구현체로 사용

주문과 할인 도메인 테스트

package hello.core.order;

import hello.core.member.Grade;
import hello.core.member.Member;
import hello.core.member.MemberService;
import hello.core.member.MemberServiceImpl;
import org.assertj.core.api.Assertions;
import org.junit.jupiter.api.Test;

public class OrderServiceTest {
    MemberService memberService = new MemberServiceImpl();
    OrderService orderService = new OrderServiceImpl();

    @Test
    void createOrder() {
        Long memberId = 1L;
        Member member = new Member(memberId, "memberA", Grade.VIP);
        memberService.join(member);

        Order order = orderService.createOrder(memberId, "itemA", 10000);

        Assertions.assertThat(order.getDiscountPrice()).isEqualTo(1000);
    }
}

0개의 댓글