[스프링 핵심 원리 - 기본편] 02. 스프링 핵심 원리 이해1 - 예제 만들기

Turtle·2024년 6월 11일
0
post-thumbnail

🧱비즈니스 요구사항과 설계

  • 회원
    - 회원을 가입하고 조회할 수 있다.
    - 회원은 일반과 VIP 두 가지 등급이 있다.
    - 회원 데이터는 자체 DB를 구축할 수 있고, 외부 시스템과 연동할 수 있다.
  • 주문과 할인 정책
    - 회원은 상품을 주문할 수 있다.
    - 회원 등급에 따라 할인 정책을 적용할 수 있다.
    - 할인 정책은 모든 VIP는 1000원을 할인해주는 고정 금액 할인을 적용해달라.
    - 할인 정책은 변경 가능성이 높다. 회사 기본 할인 정책을 아직 정하지 못했고 오픈 직전까지 고민을 미루고 싶다. 최악의 경우 할인을 적용하지 않을 수도 있다.

🔨회원 도메인 설계

  • 회원 도메인 요구사항
    - 회원을 가입하고 조회할 수 있다.
    - 회원은 일반과 VIP 두 가지 등급이 있다.
    - 회원 데이터는 자체 DB를 구축할 수 있고, 외부 시스템과 연동할 수 있다.

🔨회원 도메인 개발

회원 도메인

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;
    }
}

회원 서비스 역할

package hello.core.member;

public interface MemberService {
    void join(Member member);
    Member findMember(Long id);
}

회원 서비스 구현체

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 id) {
        return memberRepository.findById(id);
    }
}

회원 리포지토리 역할

package hello.core.member;

public interface MemberRepository {
    void save(Member member);
    Member findById(Long id);
}

회원 리포지토리 구현체

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 id) {
        Member member = store.get(id);
        return member;
    }
}

❔️static과 final

private static Map<Long, Member> store = new HashMap<>();

회원 정보를 저장하는 부분에서 Map을 사용하였고 이 때, static으로 생성하였다. 이렇게 되면 서비스에서 new MemberRepository()를 하든, 컨트롤러에서 new MemberRepository()를 하든, 테스트에서 new MemberRepository()를 하든 어플리케이션이 시작되고 종료될 때까지 Map은 오직 단 한 번만 생성된다. 그리고 이 Map을 모든 인스턴스가 공동으로 사용하게 된다.

private final MemberRepository memberRepository = new MemoryMemberRepository();

문법을 통해서도 공부했지만 자바에서는 final 키워드가 존재한다. 불변이라는 의미에 꽂힌 나머지 잘못된 개념으로 알고 있었는데 이번 기회에 확실히 짚고 넘어갈 수 있게 되었다. final 키워드는 변수의 참조를 변경하지 못하도록 하는 것이지, 해당 참조가 가리키는 객체의 내부 상태를 변경하지 못하도록 하는 것이 아니다. 즉, final 키워드가 붙은 이 memberRepository 변수는 다른 MemberRepository의 인스턴스로 변경할 수 없지만 memberRepository가 가리키는 MemberRepository의 객체의 메서드를 호출하여 데이터를 추가하거나 변경하는 것은 가능하다는 소리다.

이 final 키워드를 사용함으로써, 이 변수에 다른 객체를 할당하지 못하게 하여 코드의 안정성을 높일 수 있게 된다. 즉, 여기서는 한 번 할당된 리포지토리 객체가 프로그램 실행 중에 변경되지 않도록 보장한다는 뜻이 된다.

private static final Map<long, item> store = new HashMap<>();

변수 store가 선언될 때 값이 할당되고 이후에는 변경할 수 없다. 따라서 이 변수는 상수로 사용되고 일반적으로 변경되지 않는 값을 저장할 때 사용된다.

✅회원 도메인 실행과 테스트

  1. main 메서드를 실행시켜서 확인
package hello.core;

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

public class MemberApp {
    public static void main(String[] args) {
        MemberServiceImpl memberService = new MemberServiceImpl();
        Member member = new Member(1L, "memberA", Grade.VIP);
        memberService.join(member);

        Member findMember = memberService.findMember(1L);
        System.out.println("findMember = " + findMember.getName());
        System.out.println("member = " + member.getName());
    }
}
  1. 테스트 코드를 작성하여 확인
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() {
        Member member = new Member(1L, "memberA", Grade.VIP);

        memberService.join(member);
        Member findMember = memberService.findMember(member.getId());
        Assertions.assertThat(member.getId()).isEqualTo(findMember.getId());
    }
}

🔨주문과 할인 도메인 설계

  • 주문과 할인 정책
    - 회원은 상품을 주문할 수 있다.
    - 회원 등급에 따라 할인 정책을 적용할 수 있다.
    - 할인 정책은 모든 VIP는 1000원을 할인해주는 고정 금액 할인을 적용해달라.
    - 할인 정책은 변경 가능성이 높다. 회사 기본 할인 정책을 아직 정하지 못했고 오픈 직전까지 고민을 미루고 싶다. 최악의 경우 할인을 적용하지 않을 수도 있다.

  1. 주문 생성 : 클라이언트는 주문 서비스에 주문 생성을 요청한다.
  2. 회원 조회 : 할인을 위해서 회원 등급이 필요하다. 주문 서비스는 회원 저장소에서 회원을 조회한다.
  3. 할인 적용 : 주문 서비스는 회원 등급에 따른 할인 여부를 할인 정책에 위임한다.
  4. 주문 결과 반환 : 주문 서비스는 할인 결과를 포함한 주문 결과를 반환한다.

역할과 구현을 분리

  • 주문 서비스의 역할
    • 주문 서비스 구현체
  • 회원 저장소의 역할
    • 메모리 회원 저장소 구현체
    • DB 회원 저장소 구현체
  • 할인 정책 역할
    • 정액 할인 정책 구현체
    • 정률 할인 정책 구현체

🔨주문과 할인 도메인 개발

주문 도메인

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 findMember = memberRepository.findById(memberId);
        int discountPrice = discountPolicy.discount(findMember, itemPrice);
        return new Order(memberId, itemName, itemPrice, discountPrice);
    }
}

할인 역할

package hello.core.discount;

import hello.core.member.*;

public interface DiscountPolicy {
    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 discountAmount = 1000;

    @Override
    public int discount(Member member, int price) {
        if (member.getGrade() == Grade.VIP) {
            return discountAmount;
        } else {
            return 0;
        }
    }
}

✅주문 도메인 실행과 테스트

  1. main 메서드를 실행시켜서 확인
package hello.core;

import hello.core.order.Order;
import hello.core.order.OrderService;
import hello.core.order.OrderServiceImpl;
import hello.core.member.*;

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

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

        Order order = orderService.createOrder(1L, "itemA", 10000);
        System.out.println("order = " + order);
    }
}
  1. 테스트 코드를 작성하여 확인
package hello.core.order;

import hello.core.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() {
        Member member = new Member(1L, "memberA", Grade.VIP);
        memberService.join(member);

        Order order = orderService.createOrder(member.getId(), "itemA", 10000);
        Assertions.assertThat(order.getDiscountPrice()).isEqualTo(1000);
    }
}

🔒출처

스프링 핵심 원리 - 기본편

0개의 댓글