mojo's Blog

스프링 핵심 원리 이해 - 예제 만들기 본문

Spring

스프링 핵심 원리 이해 - 예제 만들기

_mojo_ 2022. 1. 23. 17:41

회원 설계

 

※ 회원

 

  • 회원을 가입하고 조회할 수 있다.
  • 회원은 일반과 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
Comments