(인프런) 김영한님의 스프링 핵심 원리-기본편을 공부하고 리뷰한 글입니다.
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);
}
}
'Spring > 스프링 핵심 원리 - 기본편' 카테고리의 다른 글
[스프링 핵심 원리] 03.스프링 핵심 원리 이해2(객체 지향 원리 적용) - 관심사의 분리 & AppConfig 리팩터링 (0) | 2022.05.13 |
---|---|
[스프링 핵심 원리] 03.스프링 핵심 원리 이해2(객체 지향 원리 적용) - 새로운 할인 정책 개발, 새로운 할인 정책 적용과 문제점 (0) | 2022.05.13 |
[스프링 핵심 원리] 02.스프링 핵심 원리 이해1(예제 만들기) - 회원 도메인 설계 & 회원 도메인 개발 & 회원 도메인 실행과 테스트 (0) | 2022.05.13 |
[스프링 핵심 원리] 02.스프링 핵심 원리 이해1(예제 만들기) - 프로젝트 생성 & 비즈니스 요구사항과 설계 (0) | 2022.05.13 |
[스프링 핵심 원리] 01. 객체 지향 설계와 스프링 (0) | 2022.05.13 |