💡 회원 리포지토리와 도메인을 활용해서 실제 비즈니스 로직을 작성하는 단계
service 패키지 생성 ( main > java > hello.hellospring.service ) service 패키지 아래 MemberService 클래스 생성
public class MemberService {
private final MemberRepository memberRepository = new MemoryMemberRepository();
// 회원가입
public Long join(Member member){
// 같은 이름이 있는 경우 가입X
/*
Optional<Member> result = memberRepository.findByName(member.getName());
// result.orElseGet(); Optional로 받아서 이와 같은 메소드를 사용할 수 있음
result.ifPresent( m -> { // ifPresent : null이 아닌 값이 있으면 동작
throw new IllegalStateException("이미 존재하는 회원입니다.");
});
*/
// Optional을 적지 않고 바로 반환하는 방법을 사용
// 이렇게 로직이 나오는 경우 method로 만드는 게 좋음(단축키 : ctrl+T > Extract Method)
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);
}
}
💡 cmd+option+v : return 문장을 바로 만들어주는 단축키
💡 ctrl+T > Extract Method : method 생성 단축키
💡 서비스는 보통 비즈니스에 의존적으로 설계&네이밍
💡 리포지토리는 단순히 데이터를 처리하는 느낌으로 좀 더 기계적&개발스러운 용어들을 사용
MemberService 변경 전 : MemberService가 리포지토리를 직접 생성public class MemberService {
private final MemberRepository memberRepository = new MemoryMemberRepository();
}
MemberService 변경 후 : Dependency Injection(di) 방식(외부에서 메모리 멤버 리포지토리를 넣어주는 방식) 으로 변경public class MemberService {
private final MemberRepository memberRepository;
public MemberService(MemberRepository memberRepository) {
this.memberRepository = memberRepository;
}
}
MemberServiceTest 클래스 생성 방법service패키지 추가 후, MemberServiveTest 클래스 생성 MemberService에서 Cmd+Shift+T > Create New Test > Testing Library : JUnit5, Class name : MemberServiceTest, Method 전체 선택 > OK 를 하면 자동으로 생성됨// 테스트 method는 과감하게 한글로 바꿔도 상관없음 -> 직관성 UP
// ∵ 테스트 코드는 build 될 때 실제 코드에 포함X
class MemberServiceTest {
/*
MemberService memberService = new MemberService();
// clearStore()를 위해 memberRepository를 가져와야 함
MemoryMemberRepository memberRepository = new MemoryMemberRepository();
// 문제 : memberService와 memberRepository가 다른 repository로 동작
// 해결방안 : MemberService.java에서 레포지토리를 외부에서 넣어주는 방식으로 변환하고,
// @BeforeEach 로 동작 전, repository를 넣어줌.
*/
// Dependency Injection(di) 방식 : 외부에서 메모리 멤버 리포지토리를 넣어주는 방식
MemberService memberService;
MemoryMemberRepository memberRepository;
@BeforeEach // 테스트는 독립적으로 실행이 되어야 하기 때문에, 각각의 테스트 실행 전 repository 추가
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 = memberService.findOne(saveId).get();
assertThat(member.getName()).isEqualTo(findMember.getName());
}
@Test // 정상 플로우 테스트도 중요하지만 예외 플로우 테스트도 중요 !
public void 중복_회원_예외(){
// given
Member member1 = new Member();
member1.setName("spring");
Member member2 = new Member();
member2.setName("spring");
// when
memberService.join(member1);
// try-catch 문은 대신 assertThrows사용 가능
// memberService.join을 실행했을 때 IllegalStateException 예외가 터지면 test 성공
// assertThrows(IllegalStateException.class, () -> memberService.join(member2));
IllegalStateException e = assertThrows(IllegalStateException.class, () -> memberService.join(member2));
assertThat(e.getMessage()).isEqualTo("이미 존재하는 회원입니다.");
/*
try{
memberService.join(member2);
fail();
} catch (IllegalStateException e){
// validateDuplicateMember의 예외 메세지와 같아야 함.
assertThat(e.getMessage()).isEqualTo("이미 존재하는 회원입니다.");
}
*/
// then
}
}
💡 테스트 method는 build 될 때 실제 코드에 포함되지 않으므로 한글로 바꿔도 됨
💡 Dependency Injection(di) : 외부에서 메모리 멤버 리포지토리를 넣어주는 방식
💡 given, when, then 문법 : 상황에 따라 안 맞을 때도 있지만 초기에는 이런 방식을 익히는 걸 추천
given: 어떤 데이터 기반인지
when : 어떤 걸 실행하는지
then : 어떤 결과를 검증하는지
💡 테스트는 정상 플로우도 중요하지만 예외 플로우도 중요!
💡 ctrl + R : 이전에 실행했던 코드를 다시 실행해줌 (윈도우는 Shift+F10)
💡 cmd + shift + T : 테스트 코드 자동 생성 단축키