- 회원
- 회원을 가입하고 조회할 수 있다.
- 회원은 일반과 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;
}
}
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가 선언될 때 값이 할당되고 이후에는 변경할 수 없다. 따라서 이 변수는 상수로 사용되고 일반적으로 변경되지 않는 값을 저장할 때 사용된다.
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());
}
}
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원을 할인해주는 고정 금액 할인을 적용해달라.
- 할인 정책은 변경 가능성이 높다. 회사 기본 할인 정책을 아직 정하지 못했고 오픈 직전까지 고민을 미루고 싶다. 최악의 경우 할인을 적용하지 않을 수도 있다.
역할과 구현을 분리
주문 도메인
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;
}
}
}
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);
}
}
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);
}
}