비즈니스 요구사항 정리
- 데이터: 회원ID, 이름
- 기능: 회원등록, 조회
- 아직 데이터 저장소가 선정되지 않음 (가상의 시나리오)
일반적인 애플리케이션 계층 구조
- Controller: 웹 MVC의 컨트롤러 역할
- Service: 핵심 비즈니스 로직 구현
- Repository: 데이터베이스에 접근, 도메인 객체를 DB에 저장하고 관리
- Domain: 비즈니스 도메인 객체 예) 회원, 주문, 쿠폰 등 데이터베이스에 저장하고 관리됨.
MemberRepository
public interface MemberRepository {
Member save(Member member);
Optional<Member> findById(Long id);
Optional<Member> findByName(String name);
List<Member> findAll();
}
MemoryMemberRepository
@Repository
public class MemoryMemberRepository implements MemberRepository{
private static Map<Long, Member> store = new ConcurrentHashMap<>();
private static long sequence = 0L;
@Override
public Member save(Member member) {
member.setId(sequence);
store.put(member.getId(), member);
return member;
}
@Override
public Optional<Member> findById(Long id) {
return Optional.ofNullable(store.get(id));
}
@Override
public Optional<Member> findByName(String name) {
return store.values().stream()
.filter(member -> member.getName().equals(name))
.findAny();
}
@Override
public List<Member> findAll() {
return new ArrayList<>(store.values());
}
}
회원 리포지토리 테스트 케이스 작성
MemoryMemberRepositoryTest
class MemoryMemberRepositoryTest {
MemoryMemberRepository repository = new MemoryMemberRepository();
@Test
public void save(){
Member member = new Member();
member.setName("spring");
repository.save(member);
Member result = repository.findById(member.getId()).get();
assertThat(member).isEqualTo(result);
}
@Test
public void findByName(){
Member member1 = new Member();
member1.setName("spring1");
repository.save(member1);
Member member2 = new Member();
member2.setName("spring2");
repository.save(member2);
Member result = repository.findByName("spring1").get();
assertThat(result).isEqualTo(member1);
}
@Test
public void findAll(){
Member member1 = new Member();
member1.setName("spring1");
repository.save(member1);
Member member2 = new Member();
member2.setName("spring2");
repository.save(member2);
List<Member> result = repository.findAll();
assertThat(result.size()).isEqualTo(2);
}
}
Test 코드 작성하기
Test 코드 톺아보기
@Test
public void save(){
Member member = new Member();
member.setName("spring");
repository.save(member);
Member result = repository.findById(member.getId()).get();
assertThat(member).isEqualTo(result);
}
@Test
어노테이션을 통해 테스트 메서드 생성 가능
Assertions.aaertThat()
으로 테스트 가능.
- 보통 Assertions를 import static 한 후 assertThat을 사용함.
Test 각각이 끝나고 난 뒤 해당 데이터를 제거하기
@AfterEach
public void afterEach(){
repository.clearStore();
}
- 각각의 테스트가 끝나고 나면 해당 데이터를 지워줌.
- 순차적으로 테스트가 진행되면서, 이전 테스트의 결과가 현행 테스트의 결과에 영향을 주면 안됨.
MemberService
public class MemberService {
private final MemberRepository memberRepository = new MemoryMemberRepository();
public Long join(Member member){
validateDuplicateMember(member);
memberRepository.save(member);
return member.getId();
}
public List<Member> findMembers() {
return memberRepository.findAll();
}
public Optional<Member> findOne(Long memberId){
return memberRepository.findById(memberId);
}
private void validateDuplicateMember(Member member) {
memberRepository.findByName(member.getName())
.ifPresent(m -> {
throw new IllegalStateException("이미 존재하는 회원입니다.");
});
}
}
Optional
- null인 가능성이 있는 경우 Optional로 감싸서 사용
- Optional에 다양한 메서드가 있어 해당 매서드를 사용할 수 있다.
- 자주 사용하는 메서드
- orElseGet()
- orElseThrow()
- isPresent()
- isEmpty()
Repository, Service의 설계
- Service는 보다 비즈니스 적인 부분에 중점을 두고 설계함.
- 네이밍 룰도 비즈니스 적인 내용으로 작성함
- 예) join, findMembers 등
- 기획자와 개발자 모두 손쉽게 파악할 수 있도록.
- Repository는 개발자적인 네이밍을 잡게 됨.
MemberServiceTest
class MemberServiceTest {
MemberService memberService;
MemoryMemberRepository memberRepository;
@BeforeEach
void beforeEach(){
memberRepository = new MemoryMemberRepository();
memberService = new MemberService(memberRepository);
}
@AfterEach
void afterEach(){
memberRepository.clearStore();
}
@Test
void 회원가입() {
Member member = new Member();
member.setName("hello");
Long saveId = memberService.join(member);
Member findMember = memberService.findOne(saveId).get();
assertThat(member.getName()).isEqualTo(findMember.getName());
}
@Test
void 중복_회원_예외(){
Member member1 = new Member();
member1.setName("spring");
Member member2 = new Member();
member2.setName("spring");
memberService.join(member1);
IllegalStateException e = assertThrows(IllegalStateException.class, () -> memberService.join(member2));
assertThat(e.getMessage()).isEqualTo("이미 존재하는 회원입니다.");
}
@Test
void findMembers() {
}
@Test
void findOne() {
}
}
확실한 Test
- 테스트를 확실하게 진행하기 위해선 정상플로우의 테스트도 중요하지만, 예외 상황에서의 테스트가 정상적으로 동작하는지 확인하는 것도 반드시 필요함.
Dependency Injection - 의존성 주입
public class MemberService {
private final MemberRepository memberRepository;
public MemberService(MemberRepository memberRepository) {
this.memberRepository = memberRepository;
}
}
- Service를 만들 때 각각의 Service가 다른 repository를 참조하지 않도록 memberRepository를 직접 할당해줌.