Spring/스프링 핵심 원리 - 기본편

[스프링 핵심 원리] 02.스프링 핵심 원리 이해1(예제 만들기) - 주문과 할인 도메인 설계 & 주문과 할인 도메인 개발 & 주문과 할인 도메인 실행과 테스트

HSY_mumu 2022. 5. 13. 17:38
728x90

(인프런) 김영한님의 스프링 핵심 원리-기본편을 공부하고 리뷰한 글입니다.

1. 주문과 할인 도메인 설계

<주문과 할인 정책>

 

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

1) 주문 생성: 클라이언트는 주문 서비스에 주문 생성을 요청한다.

- 실무에서는 상품이라는 객체를 만들어서 구현하지만 여기서는 간단하게 구현하기 위해 데이터(회원 id, 상품명, 상품가격)로 보낸다.

2) 회원 조회: 할인을 위해서 회원 등급이 필요하다. 그래서 주문 서비스는 회원 저장소에서 회원을 조회한다. (회원의 등급을 알기 위해서)

3) 할인 적용: 주문 서비스는 회원 등급에 따른 할인 여부를 할인 정책에 위임한다.

4) 주문 결과 반환: 주문 서비스는 할인 결과를 포함한 주문 결과를 반환한다.

(참고) 실제로는 주문 데이터를 DB에 저장하지만 여기서는 단순히 주문 결과를 반환하도록 구현한다.

 

2. 주문 도메인 전체

역할과 구현을 분리해서 자유롭게 구현 객체를 조립할 수 있도록 설계했다. 덕분에 회원 저장소, 할인 정책유연하게 변경할 수 있다. 

 

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

실제로 어떤 인터페이스 구현체로 new를 하는지에 따라서 주문 도메인 객체 다이어그램은 2가지로 나타낼 수 있다. 역할들의 협력 관계를 그대로 재사용할 수 있다.

4. 주문 도메인 객체 다이어그램1

회원을 메모리에서 조회하고 정액 할인 정책(고정금액)을 지원해도 주문 서비스를 변경하지 않아도 된다.

 

5. 주문 도메인 객체 다이어그램2

회원을 DB에서 조회하고 정률 할인 정책(주문 금액 %)을 지원해도 주문 서비스를 변경하지 않아도 된다.

 

2. 주문과 할인 도메인 개발

1. 할인 정책 인터페이스

package hello.core.discount;

import hello.core.member.Member;

public interface DiscountPolicy {
    /**
     * @return 할인 대상 금액
     */
    int discount(Member member, int price);
}

 

2. 정액 할인 정책 구현체

DiscountPolicy 인터페이스를 구현한 FixDiscountPolicy 클래스다.

회원의 등급이 VIP일 때 1000원 할인, 아니면 할인X 

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) {
        // enum 타입은 == 으로 비교
        if (member.getGrade() == Grade.VIP) {
            return discountFixAmount;
        } else {
            return 0;
        }
    }
}
  • enum 타입 == 으로 비교

3. 주문 엔티티

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;
    }
    
    // getter and setter 생략

    // toString() 재정의
    @Override
    public String toString() {
        return "Order{" +
                "memberId=" + memberId +
                ", itemName='" + itemName + '\'' +
                ", itemPrice=" + itemPrice +
                ", discountPrice=" + discountPrice +
                '}';
    }
}
  • Order 객체의 상태를 보기위해 toString() 재정의(Order 객체를 출력할 때 toString()이 자동 호출됨)

4. 주문 서비스 인터페이스

원래는 상품 객체를 생성해서 createOrder() 매개변수로 상품 객체로 넘겨야하지만 간단하게 구현하기 위해 데이터를 그냥 넘기는 방식으로 구현함

package hello.core.order;

public interface OrderService {
    // 주문 생성 메서드
    Order createOrder(Long memberId, String itemName, int itemPrice);
}

 

5. 주문 서비스 구현체

주문 생성 요청 -> 회원 정보를 조회 -> 할인 정책 적용 -> 주문 객체 생성하여 반환

메모리 회원 리포지토리 & 고정 금액 할인 정책 구현체로 생성

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{

    // MemoryMemberRepository & FixDiscountPolicy 를 구현체로 생성
    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);            // 해당 id를 가진 회원 조회
        int discountPrice = discountPolicy.discount(member, itemPrice); // 회원의 등급에 따른 할인 금액

        return new Order(memberId, itemName, itemPrice, discountPrice); // 최종 생성된 주문 반환
    }
}

 

3. 주문과 할인 도메인 실행과 테스트

1. 주문과 할인 정책 실행

애플리케이션 로직으로 아래와 같이 main()에서 매번 테스트 하는 것은 좋은 방법이 아니다. JUnit 테스트를 활용하자.

package hello.core;

import hello.core.member.Grade;
import hello.core.member.Member;
import hello.core.member.MemberService;
import hello.core.member.MemberServiceImpl;
import hello.core.order.Order;
import hello.core.order.OrderService;
import hello.core.order.OrderServiceImpl;

public class OrderApp {
    public static void main(String[] args) {
        MemberService memberService = new MemberServiceImpl();
        OrderService orderService = new OrderServiceImpl();

        Long memberId = 1L;
        Member member = new Member(memberId, "memberA", Grade.VIP);
        memberService.join(member);

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

        System.out.println("order = " + order); // order.toString() 출력됨
        //System.out.println("order.calculatePrice = " + order.calculatePrice()); // 할인된 최종 가격
    }
}

<결과>

 

2. 주문과 할인 정책 테스트(JUnit 활용) 

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);
    }
}
728x90