1. 비즈니스 요구사항과 설계
- 회원
- 회원 가입 및 조회
- 회원 등급 일반, vip 등급
- 회원 데이터는 자체 DB 구축, 외부 시스템과 연동가능(미확정)
- 주문과 할인 정책
- 회원은 상품주문 가능
- 등급에 따른 할인 정책 적용
- 할인 정책은 모든 vip는 1000원을 할인해주는 고정 금액 할인(나중에 변경될수 있음)
- 할인 정책은 변경 가능성이 높아서 오픈 직전까지 고민을 미루고 싶으며, 할인을 적용하지 않을 수 도 있다.
→ 인터페이스를 만들고 구현체를 언제든지 갈아 끼울수 있도록 설계 한다.
2. 회원 도메인 설계
- 회원 도메인 요구 사항
- 가입 및 조회 기능
- 일반과 vip 두가지 등급
- 회원 데이터는 자체 DB구축할수 있고, 외부 시스템과 연동 가능(미확정)
→ 어떤 DB를 사용할지 미확정이기 때문에 회원 저장소 인터페이스를 만들고 메모리 회원 저장소를 이용해서 기능을 구현해본다.
- 멤버 서비스 인터페이스와 서비스 회원 저장소인 MemberRepository 인터페이스를 만들어서 둘이 연관 시킨다.
- Repository의 구현 객체를 갈아 끼우기 가능하게 설계
3. 회원 도메인 개발
3-1 Grade Enum생성
package hello.core.member;
public enum Grade {
BASIC,
VIP
}
Enum으로 BASIC과 VIP 등급을 생성
3-2 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;
}
}
Member객체는 id, name, grade 멤버변수를 가지고 각각 getter와 setter 생성
3-3 회원 DB 인터페이스 생성
package hello.core.member;
public interface MemberRepository {
void save(Member member);
Member findById(Long memberId);
}
가입하려는 member객체를 저장하는 save와 조회 요청이 오면 찾는 findById 기능으로 구성
3-4 회원 서비스 인터페이스 생성
package hello.core.member;
public interface MemberService {
void join(Member member);
Member findMember(Long memberId);
}
가입과 조회 메서드를 가지게 생성
3-5 회원 서비스 구현 객체 생성
package hello.core.member;
public class MemberServiceImpl implements MembertService{
private final MemberRepository memberRepository = new MemoryMemberRepository();
@Override
public void join(Member member) {
}
@Override
public Member findMember(Long memberId) {
return memberRepository.findById(memberId);
}
}
가입과 조회 기능을 수행하는 객체 클래스
3-6 회원DB를 Memory상에서 구현하는 객체 생성
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);
}
}
MemoryMemberRepository의 Map에 Member객체를 저장하는 기능과 조회하는 기능을 가지는 DB 생성
4. 회원 도메인 가입 테스트
테스트 폴더에 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);
}
}
@Test 어노테이션 기입
Member 객체 생성후, memverService의 findMember를 이용해서 member와 findMember가 같은지 비교
회원 도메인 설계의 문제점
- 의존 관계가 인터페이스 뿐만 아니라 구현까지 모두 의존하는 문제점이 있다.
public class MemberServiceImpl implements MemberService {
private final MemberRepository memberRepository = new MemoryMemberRepository();
//MemberServiceImple는 MemberRepository를 의존한다. 문제는 MemmoryMemberRepository도 의존한다.
@Override
public void join(Member member) {
memberRepository.save(member);
}
@Override
public Member findMember(Long memberId) {
return memberRepository.findById(memberId);
}
}
MemberRepository → 인터페이스 : 추상화
MemmoryMemberRepository → 클래스 : 구체화
현재 MemberServiceImpl은 추상화에도 의존하고, 구체화에도 의존하는 문제점이 있다. → DIP를 위반하고 있음
반응형