유정잉

🎀 스프링 기본 1 본문

개발자 공부/🎀 스프링 공부

🎀 스프링 기본 1

유정♡ 2024. 5. 23. 17:31

 

[ 비즈니스 요구사항과 설계 ]

    - 회원

      회원을 가입하고 조회할 수 있다.

      회원은 일반과 VIP 두 가지 등급이 있다.

      회원 데이터는 자체 DB를 구축할 수 있고, 외부 시스템과 연동할 수 있다. (미확정)

   - 주문과 할인 정책

     회원은 상품을 주문할 수 있다.

     회원 등급에 따라 할인 정책을 적용할 수 있다.

      할인 정책은 모든 VIP1000원을 할인해주는 고정 금액 할인을 적용. (나중에 변경 될 수 있다.) 할인 정책은 변경 가능성이 높다.

      회사의 기본 할인 정책을 아직 정하지 못했고, 오픈 직전까지 고민을 미루고싶다. 최악의 경우 할인을 적용하지 않을 수 있다. (미확정)

 

요구사항을 보면 회원 데이터, 할인 정책 같은 부분은 지금 결정하기 어려운 부분이다. 그렇다고 이런 정책이 결정될 때 까지 개발을 무기한 기다릴 수 도 없다. 우리는 앞에서 배운 객체 지향 설계 방법이 있지 않은가!

🌷 인터페이스를 만들고 구현체를 언제든지 갈아끼울 수 있도록 설계하면 된다 !!!

메모리는 로컬(=서버를 재부팅 하면 정보가 날아감)


1) member 패키지 생성 -> Grade는 Enum으로 VIP등급 설정, Member는 class로 id, name, grade 생성

package hello.core.member;

public enum Grade {
    BASIC,
    VIP
}
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;
    }
}

 

2) Interface MemberRepository 생성(Interface만으로는 작동하지 않으므로 회원 저장소를 만든후 구현체도 만들어 줘야 함 !!)

     -> class MemoryMemberRepository 생성

     이때 HashMap을 사용하는데 !! 실무에서는 동시성 이슈 때문에 Concurrent HashMap을 써야한다 !!

package hello.core.member;

//회원 저장소를 만들었으니 구현체도 만들어 줘야 함 !! Interface로만 작동하지 않음
public interface MemberRepository {

    void save(Member member); //회원 정보 저장

    Member findById(Long memberId); //회원의 Id로 회원을 찾는 기능
}
package hello.core.member;

import java.util.HashMap;
import java.util.Map;

public class MemoryMemberRepository implements MemberRepository {

    //실무에서는 동시성 이슈가 발생할 수 있어서 Concurrent HashMap을 써야함(그치만 복잡하기 떄문에 공부할때는 그냥 HashMap 사용)
    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); //findById호출하면 store에서 get해서 memberId널으면 바로 Id로 찾는 것
    }
}

 

3) Interface MemberService 생성 후 -> MemberServiceImpl 생성(보통 구현체가 하나만 나오면 클래스명 뒤에 Impl 붙여 줌)

package hello.core.member;

public interface MemberService {

    void join(Member member);

    Member findMember(Long memberId);

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

4) 지금까지 만든 내용을 순수한 자바 코드로(=스프링없이) 확인해 보는 방법 ! (=좋지 않은 방법) -> MemberApp 클래스 생성 

package hello.core;

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

public class MemberApp {

    public static void main(String[] args) {
        MemberService memberService = new MemberServiceImpl();
        Member member = new Member(1L, "memberA", Grade.VIP); // 🩷 command + option + v = Member member 생성해줌
        memberService.join(member); //member를 넣으면 join이 되야함

        Member findMember = memberService.findMember(1L);
        System.out.println("new member = " + member.getName());
        System.out.println("find member = " + findMember.getName());
    }
}

 

5) 순수 자바 코드로 확인하는 방법 대신 @Test를 활용하여 확인 하는 방법 !! (⭐️중요) -> 녹색 체크가 뜨면 제대로 만든 것 

package hello.core.member;

import org.assertj.core.api.Assertions;
import org.junit.jupiter.api.Test;

public class MemberServiceTest {

    MemberService memberService = new MemberServiceImpl(); //join함수 쓰기 위해서

    @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); //member와 findMember와 똑같냐
    }
}

 

[ 회원 도메인 설계의 문제점 ]

다른 저장소로 변경할 때 OCP 원칙을 잘 준수할까요? DIP를 잘 지키고 있을까요?
**의존관계가 인터페이스 뿐만 아니라 구현까지 모두 의존하는 문제점이 있음**

**주문까지 만들고나서 문제점과 해결 방안을 설명**

 


[ 주문과 할인 도메인 설계 ]

회원은 상품을 주문할 수 있다.

회원 등급에 따라 할인 정책을 적용할 수 있다.

할인 정책은 모든 VIP1000원을 할인해주는 고정 금액 할인을 적용해달라. (나중에 변경 될 수 있다.)

할인 정책은 변경 가능성이 높다. 회사의 기본 할인 정책을 아직 정하지 못했고, 오픈 직전까지 고민을 미루고 싶다.

최악의 경우 할인을 적용하지 않을 수 도 있다. (미확정)

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

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

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

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

 

1) discount 패키지 생성 -> Interface DiscountPolicy 생성 -> FixDiscountPolicy class 생성 구현

package hello.core.discount;

import hello.core.member.Member;

public interface DiscountPolicy {

    // @return 할인 대상 금액
    int discount(Member member, int price);
}
package hello.core.discount;

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

public class FixDincountPolicy implements DiscountPolicy {

    private int discountFixAmount = 1000; //1000원 할인

    @Override
    public int discount(Member member, int price) {
        if (member.getGrade() == Grade.VIP) { //Enum은 == 사용함
            return discountFixAmount;
        } else {
            return 0;
        }
    }
}

 

2) order 패키지 생성 -> Order class 생성 -> Interface OrderSerivce 생성 -> OrderServiceImple class 생성

    Order에는 할인에 대한 비즈니스 로직과, 출력할때 보기 쉽게 하기 위한 ToString 사용

    이때 OrderService는 최종 주문 결과를 반환하기 위한 Interface

    OrderServiceImple은 주문 요청이 오면 회원 정보를 조회 후 할인 정책에 회원정보와 가격을 넘긴 후 최종 생성된 주문 반환

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

    //출력할때 보기 쉽게 하려고 ToString
    @Override
    public String toString() {
        return "Order{" +
                "memberId=" + memberId +
                ", itemName='" + itemName + '\'' +
                ", itemPrice=" + itemPrice +
                ", discountPrice=" + discountPrice +
                '}';
    }
}
package hello.core.order;

//최종 Order 결과를 반환하기 위한 Interface 역할
public interface OrderService {
    Order createOrder(Long memberId, String itemName, int itemPrice);
}
package hello.core.order;

import hello.core.discount.DiscountPolicy;
import hello.core.discount.FixDincountPolicy;
import hello.core.member.Member;
import hello.core.member.MemberRepository;
import hello.core.member.MemoryMemberRepository;

//주문 생성 요청이 오면 먼저 회원 정보를 조회를 하고 할인 정책에 회원과 가격을 넘김 그리고 최종 생성된 주문을 반환
public class OrderServiceImple implements OrderService{

    private  final MemberRepository memberRepository = new MemoryMemberRepository();
    private final DiscountPolicy discountPolicy = new FixDincountPolicy();

    //단일 책임 원칙 : 할인에 대한 부분은 discountPolicy이 알아서 해줘 ! 나는 상관 없서 !
    @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);
    }
}

 

3) OrderApp 패키지 생성 후 순수 자바 코드로 테스트 해보기 

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.OrderServiceImple;

public class OrderApp {
    public static void main(String[] args) {
        
        MemberService memberService = new MemberServiceImpl();
        OrderService orderService = new OrderServiceImple();
        
        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);
    }
}

 

4) @Test로 해보기 

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 OrderServiceImple();
    
    @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);
    }
}

 


[ 객체 지향 원리 적용 ] 

     새로운 할인 정책 개발 -> 할인율만 바꿔주는 작업을 하면 됨 ! 

 

1) RateDiscountPolicy 클래스 생성

package hello.core.discount;

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

import java.util.GregorianCalendar;

//@Test
public class RateDiscountPolicy implements DiscountPolicy{

    private int discountPercent = 10;

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

 

2) 🩷 commant + shift + T = @Test 자동 생성 => Testing library : JUnit5, Class name : class명 뒤에 Test

package hello.core.discount;

import hello.core.member.Grade;
import hello.core.member.Member;
import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Test;

import static org.assertj.core.api.Assertions.assertThat;
import static org.junit.jupiter.api.Assertions.*;

class RateDiscountPolicyTest {

    RateDiscountPolicy discountPolicy = new RateDiscountPolicy();

    @Test
    @DisplayName("VIP는 10% 할인이 적용되어야 한다.")  void vip_o() {
        //given
        Member member = new Member(1L, "memberVIP", Grade.VIP);

        //when
        int discount = discountPolicy.discount(member, 10000);

        //then
        assertThat(discount).isEqualTo(1000);
    }

    @Test
    @DisplayName("VIP가 아니면 할인이 적용되지 않아야 한다.") void vip_x() {
        //given
        Member member = new Member(2L, "memberBASIC", Grade.BASIC);

        //when
        int discount = discountPolicy.discount(member, 10000);

        //then
        assertThat(discount).isEqualTo(0);
    }

}

 

3) AppConfig는 애플리케이션의 실제 동작에 필요한 구현 객체를 생성한다

    AppConfig는 생성한 객체 인스턴스의 참조(레퍼런스)성자를 통해서 주입(연결)해준다.

    즉, MemberServiceImpl에는 MemeryMemberRepository가 들어가고 

         OrderServiceimpl에는 MemoryMemberRepository와 FixDiscountPolicy가 들어간다 

package hello.core;

import hello.core.discount.FixDincountPolicy;
import hello.core.member.MemberService;
import hello.core.member.MemberServiceImpl;
import hello.core.member.MemoryMemberRepository;
import hello.core.order.OrderService;
import hello.core.order.OrderServiceImple;

public class AppConfig {

    public MemberService memberService() {
        //MemberServiceImpl 구현체인 객체가 생성되는데 그때 MemoryMemberRepository가 들어감
        return  new MemberServiceImpl(new MemoryMemberRepository());
    }

    public OrderService orderService() {
        return  new OrderServiceImple(new MemoryMemberRepository(), new FixDincountPolicy());
    }
}
package hello.core.member;

public class MemberServiceImpl implements MemberService {

    private final MemberRepository memberRepository;

    //생성자를 통해서 memberRepository에 구현체가 뭐가 들어갈지 생성자를 통해서 선택
    public MemberServiceImpl(MemberRepository memberRepository) {
        this.memberRepository = memberRepository;
    }

    @Override
    public void join(Member member) {
        memberRepository.save(member);
    }

    @Override
    public Member findMember(Long memberId) {
        return memberRepository.findById(memberId);
    }
}
package hello.core.order;

import hello.core.discount.DiscountPolicy;
import hello.core.member.Member;
import hello.core.member.MemberRepository;
import hello.core.member.MemoryMemberRepository;

//주문 생성 요청이 오면 먼저 회원 정보를 조회를 하고 할인 정책에 회원과 가격을 넘김 그리고 최종 생성된 주문을 반환
public class OrderServiceImple implements OrderService{

    private  final MemberRepository memberRepository; //final이 있으면 무조건 생성자 있어야 함
    private final DiscountPolicy discountPolicy; //구체에 의존하지 않고 Interface에만 의존

    public OrderServiceImple(MemberRepository memberRepository, DiscountPolicy discountPolicy) {
        this.memberRepository = memberRepository;
        this.discountPolicy = discountPolicy;
    }

    //단일 책임 원칙 : 할인에 대한 부분은 discountPolicy이 알아서 해줘 ! 나는 상관 없서 !
    @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);
    }
}

-> 이렇게 함으로써 MemberServiceImpl은 MemoryMemberRepository를 의존하지 않고 단지 MemberRepository 인터페이스만 의존한다 !!! 이렇게 되면서 MemberServiceImpl은 생정자를 통해 어떤 구현 객체가 주입될지 알 수 없다. 어떤 구현 객체를 주입할지는 오직 AppConfig에서 결정된다. MemberServiceImpl은 의존관계에 대한 고민은 외부(=AppConfig)에 맡기고 실행에만 집중한다.

-> 객체를 생성하고 연결하는 역할(=AppConfig)과 실행하는 역할이 명확히 분리 됨 -> DIP 완성 

 

-> OrderServiceImpl도 FixDiscountPolicy에 의존하지 않고 단지 DiscountPolicy 인터페이스만 의존한다. OrderServiceImpl입장에서는 생성자를 통해 어떤 구현 객체가 주입될지 알 수 없다 OrderServiceImpl 실행에만 집중하면 된다  

 

4) MemberApp과 OrderApp에 AppConfig 등록 -> Interface에만 의존 더 이상 구체 클래스에 의존할 필요가 없음 

package hello.core;

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

public class MemberApp {

    public static void main(String[] args) {

        AppConfig appConfig = new AppConfig();
        MemberService memberService = appConfig.memberService(); //MemberServiceImpl이 들어감 
       // MemberService memberService = new MemberServiceImpl();
        
        Member member = new Member(1L, "memberA", Grade.VIP); // 🩷 command + option + v = Member member 생성해줌
        memberService.join(member); //member를 넣으면 join이 되야함

        Member findMember = memberService.findMember(1L);
        System.out.println("new member = " + member.getName());
        System.out.println("find member = " + findMember.getName());
    }
}
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.OrderServiceImple;

public class OrderApp {
    public static void main(String[] args) {

        AppConfig appConfig = new AppConfig();
        MemberService memberService = appConfig.memberService();
        OrderService orderService = appConfig.orderService();
        
        //MemberService memberService = new MemberServiceImpl();
        //OrderService orderService = new OrderServiceImple();

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

 

5) Test코드도 수정 

package hello.core.member;

import hello.core.AppConfig;
import org.assertj.core.api.Assertions;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;

public class MemberServiceTest {

    MemberService memberService;
    
    @BeforeEach //각 테스트 실행 전에 무조건 실행 됨
    public void beforeEach() {
        AppConfig appConfig = new AppConfig();
        memberService = appConfig.memberService();
    }
    //MemberService memberService = new MemberServiceImpl(); //join함수 쓰기 위해서

    @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); //member와 findMember와 똑같냐
    }
}
package hello.core.order;

import hello.core.AppConfig;
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.BeforeEach;
import org.junit.jupiter.api.Test;

public class OrderServiceTest {

    MemberService memberService;
    OrderService orderService;

    @BeforeEach
    public void beforeEach() {
        AppConfig appConfig = new AppConfig();
        memberService = appConfig.memberService();
        orderService = appConfig.orderService();
    }

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

 

 


[ AppConfig 구체화 ]

package hello.core;

import hello.core.discount.DiscountPolicy;
import hello.core.discount.FixDiscountPolicy;
import hello.core.discount.RateDiscountPolicy;
import hello.core.member.MemberRepository;
import hello.core.member.MemberService;
import hello.core.member.MemberServiceImpl;
import hello.core.member.MemoryMemberRepository;
import hello.core.order.OrderService;
import hello.core.order.OrderServiceImple;

public class AppConfig {

    public MemberService memberService() {
        return new MemberServiceImpl(memberRepository());
    }

    public MemberRepository memberRepository() {
        return new MemoryMemberRepository();
    }

    public OrderService orderService() {
        return new OrderServiceImple(memberRepository(), discountPolicy());
    }

    public DiscountPolicy discountPolicy() {
        return new RateDiscountPolicy();
        //FixDiscountPolicy를 RateDiscountPolicy로 바꿈
    }
}

 

[ FixDiscountPolicy를 RateDiscountPolicy로 바꿀려면 이제는 AppConfig만 변경하면 된다 ! ]

구성 영역만(=AppConfig) 변경하면 된다 ! 사용 영역은(=~Impl) 전혀 영향이 없다 !

 

🌷 역할과 구현을 명확하게 분리하자 !!!

 

 

 

 

 

 

 

 

728x90