비즈니스 요구사항 정리
- 데이터 : 회원 ID, 이름
- 기능 : 회원등록, 회원조회
- 아직 데이터 저장소가 선정되지 않음(가상의 시나리오)
일반적인 웹 어플리케이션 계층 구조

- 컨트롤러 : 웹 MVC의 컨트롤러 역할
- 서비스 : 핵심 비즈니스 로직 구현
- 리포지토리 : 데이터베이스에 접근, 도메인 객체를 DB에 저장하고 관리
- 도메인 : 비즈니스 도메인 객체, 예) 회원, 주문, 쿠폰 등을 주로 데이터베이스에 저장하고 관리됨
클래스 의존 관계

회원객체
package hello.hellospring.domain;
public class Member {
private long id;
private String name;
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;
}
}
회원 리포지토리 인터페이스
package hello.hellospring.repository;
import hello.hellospring.domain.Member;
import java.util.List;
import java.util.Optional;
public interface MemberRepository {
Member save(Member member);
Optional<Member> findById(Long id);
Optional<Member> findByName(String name);
List<Member> findAll();
}
회원 리포지토리 메모리 구현체
package hello.hellospring.repository;
import hello.hellospring.domain.Member;
import java.util.*;
public class MemoryMemberRepository implements MemberRepository {
private static Map<Long, Member> store = new HashMap<>();
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());
}
public void clearStore() {
store.clear();
}
}
회원 리포지토리 테스트 케이스 작성
개발한 기능을 실행해서 테스트 할 때 자바의 main 메서드를 통해서 실행하거나
웹 어플리케이션 컨트롤러를 통해서 해당 기능을 실행한다
이런 방법들은 준비하고 실행하는데 오래걸리고 반복 실행이 어렵고
여러 테스트를 한번에 실행하기 어렵다는 단점이있다
그래서 자바는 Junit이라는 프레임워크로 테스트를 실행해서 문제를 해결한다
회원 리포지토리 메모리 구현체 테스트
@AfterEach 를 쓰는 이유는
한번에 여러 테스트를 실행하면 메모리 DB에 직전 테스트 결과가 남을 수 있다
이러면 다음 테스트가 이전 테스트 때문에 실패할 수 있다
@AfterEach를 사용하면 각 테스트가 종료될 때 마다 이 기능을 실행한다
테스트는 각각 독립적으로 실행되어야 한다
테스트 순서에 의존관계가 있는 것은 좋은 테스트가 아니다
지금은 구현을 해놓고 테스트를 했지만
TDD 방식으로 테스트케이스를 만들어두고 거기에 맞춰 구현을 하는것이 좋다
src -> test -> java -> 테스트할 클래스와 동일한 이름으로
package hello.hellospring.repository;
import hello.hellospring.domain.Member;
import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.Test;
import java.util.List;
import static org.assertj.core.api.Assertions.*;
class MemoryMemberRepositoryTest {
MemoryMemberRepository repository = new MemoryMemberRepository();
@AfterEach
public void afterEach() {
repository.clearStore();
}
@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(){
//given
Member member1 = new Member();
member1.setName("spring1");
repository.save(member1);
Member member2 = new Member();
member2.setName("spring2");
repository.save(member2);
//when
Member result = repository.findByName("spring1").get();
//then
assertThat(result).isEqualTo(member1);
}
@Test
public void findAll() {
//given
Member member1 = new Member();
member1.setName("spring1");
repository.save(member1);
Member member2 = new Member();
member2.setName("spring2");
repository.save(member2);
//when
List<Member> result = repository.findAll();
//then
assertThat(result.size()).isEqualTo(2);
}
}
회원 서비스 개발
package hello.hellospring.service;
import hello.hellospring.domain.Member;
import hello.hellospring.repository.MemberRepository;
import hello.hellospring.repository.MemoryMemberRepository;
import java.util.List;
import java.util.Optional;
public class Memberservice {
private final MemberRepository memberRepository = new MemoryMemberRepository();
//회원가입
public Long join(Member member) {
validateDuplicateMember(member); //중복 회원 검증
memberRepository.save(member);
return member.getId();
}
private void validateDuplicateMember(Member member) {
memberRepository.findByName(member.getName())
.ifPresent(m -> {
throw new IllegalStateException("이미 존재하는 회원입니다.");
});
}
//전체 회원 조회
public List<Member> findMembers() {
return memberRepository.findAll();
}
public Optional<Member> findOne(Long memberId) {
return memberRepository.findById(memberId);
}
}
회원 서비스 테스트
윈도우 기분 control + shift + T 하면 쉽게 테스트 케이스 만드는 단축기이다
빌드될때 테스트 코드는 포함되지 않기 때문에 직관적으로 보기 위해서
메서드 이름을 한글로 바꿔서 실행해도 된다
@BeforeEach 는 각 테스트 실행 전에 호출된다
테스트가 서로 영향이 없도록 항상 새로운 객체를 생성하고 의존관계도 새로 맺어준다
package hello.hellospring.service;
import hello.hellospring.domain.Member;
import hello.hellospring.repository.MemoryMemberRepository;
import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
import static org.assertj.core.api.Assertions.*;
import static org.junit.jupiter.api.Assertions.*;
class MemberserviceTest {
MemberService memberService;
MemoryMemberRepository memberRepository;
@BeforeEach
public void beforeEach() {
memberRepository = new MemoryMemberRepository();
memberService = new MemberService(memberRepository);
}
@AfterEach
public void afterEach() {
memberRepository.clearStore();
}
@Test
void 회원가입() {
//Given
Member member = new Member();
member.setName("hello");
//When
Long saveId = memberService.join(member);
//Then
Member findMember = memberRepository.findById(saveId).get();
assertEquals(member.getName(), findMember.getName());
}
@Test
void 중복_회원_예외() throws Exception {
//Given
Member member1 = new Member();
member1.setName("spring");
Member member2 = new Member();
member2.setName("spring");
//When
memberService.join(member1);
IllegalStateException e = assertThrows(IllegalStateException.class,
() -> memberService.join(member2));//예외가 발생해야 한다.
assertThat(e.getMessage()).isEqualTo("이미 존재하는 회원입니다.");
}
}
'Spring' 카테고리의 다른 글
[Spring] 스프링 DB 접근 기술 (스프링입문 / 인프런 김영한님 강의) (0) | 2022.10.01 |
---|---|
[Spring] 회원관리예제 (스프링입문 / 인프런 김영한님 강의) (0) | 2022.10.01 |
[Spring] 스프링 빈과 의존관계 (스프링입문 / 인프런 김영한님 강의) (0) | 2022.10.01 |
[Spring] 스프링 웹 개발 기초 (스프링입문 / 인프런 김영한님 강의) (0) | 2022.10.01 |
[Spring] 프로젝트 환경설정 (스프링입문 / 인프런 김영한님 강의) (0) | 2022.10.01 |