mojo's Blog
스프링 핵심 원리 이해 - 예제 만들기 본문
회원 설계
※ 회원
- 회원을 가입하고 조회할 수 있다.
- 회원은 일반과 VIP 두 가지 등급이 있다.
- 회원 데이터는 자체 DB를 구축할 수 있고, 외부 시스템과 연동할 수 있다.
회원 도메인 협력 관계
회원 클래스 다이어그램
회원 도메인 개발
[java] -> [hello] 아래에 core 디렉토리를 만들고 그 아래에 member 디렉토리를 만든다.
그리고 총 6개의 클래스 및 인터페이스 코드를 아래와 같이 작성한다.
① Grade (class)
package hello.core.member;
public enum Grade {
BASIC,
VIP
}
② Member (class)
package hello.core.member;
public class Member {
private Long id;
private String name;
private Grade grade;
public Member(Long id, String name, Grade grade) {
this.id = id;
this.name = name;
this.grade = grade;
}
public Long getId() {
return id;
}
public void setId(Long id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Grade getGrade() {
return grade;
}
public void setGrade(Grade grade) {
this.grade = grade;
}
}
- alt + Insert : Generate 를 띄워서 Constructor 및 Getter/Setter 를 자동으로 완성할 수 있다. (꿀팁)
③ MemberRepository (interface)
package hello.core.member;
public interface MemberRepository {
void save(Member member);
Member findById(Long memberId);
}
④ MemoryMemberRepository (class)
package hello.core.member;
import java.util.HashMap;
import java.util.Map;
public class MemoryMemberRepository implements MemberRepository{
private static Map<Long, Member> store = new HashMap<>();
@Override
public void save(Member member) {
store.put(member.getId(), member);
}
@Override
public Member findById(Long memberId) {
return store.get(memberId);
}
}
⑤ MemberService (interface)
package hello.core.member;
public interface MemberService {
void join(Member member);
Member findMember(Long memberId);
}
⑥ MemberServiceImpl
package hello.core.member;
public class MemberServiceImpl implements MemberService {
private final MemberRepository memberRepository = new MemoryMemberRepository();
@Override
public void join(Member member) {
memberRepository.save(member);
}
@Override
public Member findMember(Long memberId) {
return memberRepository.findById(memberId);
}
}
회원 도메인 실행과 테스트
※ 순수 자바 코드로 테스트
[hello] -> [core] -> [member] 아래에 MemberApp 클래스를 만들고 다음과 같이 코드를 작성한다.
package hello.core.member;
public class MemberApp {
// psvm : 자동으로 public static void main 생성
public static void main(String[] args) {
MemberService memberService = new MemberServiceImpl();
// ctrl + alt + v : 자동으로 Member member 생성
Member member = new Member(1L, "memberA", Grade.VIP);
memberService.join(member);
Member findMember = memberService.findMember(1L);
System.out.println("new member = " + member.getName());
System.out.println("find member = " + findMember.getName());
}
}
- psvm : 자동으로 public static void main(String[] args) { } 를 생성해준다.
- ctrl + alt + v : 자동으로 객체를 할당받을 수 있도록 Member member가 생성된다.
이번엔 [test] -> [java] -> [hello] -> [core] -> [member] 아래에 MemberServiceTest 클래스를 만든 후 다음과 같이 코드를 작성한다.
package hello.core.member;
import org.assertj.core.api.Assertions;
import org.junit.jupiter.api.Test;
public class MemberServiceTest {
MemberService memberService = new MemberServiceImpl();
@Test
void join(){
// given
Member member = new Member(1L, "memberA", Grade.VIP);
// when
memberService.join(member);
Member findMember = memberService.findMember(1L);
// then
Assertions.assertThat(member).isEqualTo(findMember);
}
}
※ 회원 도메인 설계의 문제점
OCP 원칙을 준수하지 않고 및 DIP를 지키지 않았다.
의존관계가 인터페이스 뿐만 아니라 구현까지 모두 의존하는 문제점이 있다.
주문과 할인 정책 설계
※ 주문과 할인 정책
- 회원은 상품을 주문할 수 있다.
- 회원 등급에 따라 할인 정책을 적용할 수 있다.
- 할인 정책은 모든 VIP는 1000원을 할인해주는 고정 금액 할인을 적용한다.
- 할인 정책은 변경 가능성이 높다. 회사의 기본 할인 정책을 아직 정하지 못했고, 오픈 직전까지 고민을 미룰고 싶다. 최악의 경우 할인을 적용하지 않을 수 있다.
※ 주문 도메인 협력, 역할, 책임
- 주문 생성 : 클라이언트는 주문 서비스에 주문 생성을 요청한다.
- 회원 조회 : 할인을 위해서는 회원 등급이 필요하다. 그래서 주문 서비스는 회원 저장소에서 회원을 조회한다.
- 할인 적용 : 주문 서비스는 회원 등급에 따른 할인 여부를 할인 정책에 위임한다.
- 주문 결과 반환 : 주문 서비스는 할인 결과를 포함한 주문 결과를 반환한다.
※ 주문 도메인 전체
※ 주문 도메인 클래스 다이어그램
주문과 할인 도메인 개발
[hello] -> [core] 폴더 아래에 discount 폴더와 order 폴더를 생성한다.
그리고 discount 폴더 아래에 DiscountPolicy 인터페이스와 FixDiscountPolicy 클래스를 생성하고,
order 폴더 아래에는 OrderService 인터페이스와 Order, OrderServiceImpl 클래스를 생성한다.
① DiscountPolicy (interface)
package hello.core.discount;
import hello.core.member.Member;
public interface DiscountPolicy {
/*
@return 할인 대상 금액
*/
int discount(Member member, int price);
}
② FixDiscountPolicy (class)
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;
}
}
③ OrderService (interface)
package hello.core.order;
public interface OrderService {
Order createOrder(Long memberId, String itemName, int itemPrice);
}
④ Order (class)
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 +
'}';
}
}
⑤ OrderServiceImpl (class)
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);
}
}
주문과 할인 도메인 실행과 테스트
[order] 폴더 아래에 OrderApp 클래스를 생성한 후 다음과 같이 코드를 작성하여 테스트한다.
package hello.core.order;
import hello.core.member.Grade;
import hello.core.member.Member;
import hello.core.member.MemberService;
import hello.core.member.MemberServiceImpl;
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);
System.out.println("order.calculatePrice = " + order.calculatePrice());
}
}
이번엔 [test] -> [hello] -> [core] 아래에 order 폴더를 생성한 후 OrderServiceTest 클래스를 생성하여 다음과 같이 테스트를 진행한다.
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' 카테고리의 다른 글
AppConfig 리팩터링 및 IoC, DI, 그리고 컨테이너 (0) | 2022.01.24 |
---|---|
DI(Dependency Injection) (0) | 2022.01.23 |
객체 지향 설계와 스프링 (0) | 2022.01.20 |
AOP (0) | 2022.01.15 |
JPA (0) | 2022.01.14 |