본문 바로가기

spring boot

spring boot 04 [ 회원 관리 예제 ② ]

 

 

 

[ 회원 서비스 개발 ]

 

비지니스 로직을담당하는 서비스단을 만들어보자!

일단 service 패키지에 MemberService.java파일을 생성한다.

 

 

MemberService에서 domain의 memberRepository의 값을 가져와야 하기 때문에 객체 생성해준다.

public class MemberService {

    private final MemberRepository memberRepository = new MemoryMemberRepository();

 

 

 

일단 회원가입 기능부터 구현하자

 

/*
회원가입
*/
public Long join(Member member){
    // 같은 이름이 있는 중복 회원 X
    Optional<Member> result = memberRepository.findByName(member.getName());
    result.ifPresent(m -> {
        throw new IllegalStateException("이미 존재하는 회원입니다.");
    }); 
    // 빠르게 예외처리를 할 수 있다

    memberRepository.save(member);
    return member.getId();
}

 

 

강사님은 Optional<> 의 ifPresent를 사용해서 중복값을 처리했다.

ifPresent()
- Void 타입
- ifPresent()는 Optional 객체가 값을 가지고 있으면 실행 값이 없으면 넘어간다.

 

이때 Optional<Member> 으로 반환을 받았는데, 이렇게 받는게 별로 좋지 않다고 한다. 

 

 

 memberRepository.findByName(member.getName())
               .ifPresent(m -> {
            throw new IllegalStateException("이미 존재하는 회원입니다.");
        });

 

이렇게 코드를 수정해서 바로 받을 수 있게 해준다!

그래서 다시 메서드를 나누고 수정해준 코드는 아래와같다

/*
회원가입
*/
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);
    }

 

이렇게 모든 코드를 작성했으면?! 테스트를 진행해줘야한다

앞서 controller test를 파일만들고 수기로 써서 진행했다면 이번엔 간단하게 해보자!

 

 

단축키 : Ctrl + Shift + t 

 

 

새 테스트 생성을 클릭하면 아래처럼 창이 뜨고

 

 

 

 

 

 

 

 

 

 

 

 

 

 

<< 전부 체크해준뒤 확인을 누른다

 

 

 

 

 

 

 

그러면 test폴더에 자동으로  메서드가 생성이 된다!

 

테스트는 한글로 작성해도 무방하다! (영어권 팀원과 일하지 않는이상) 

@Test
    void 회원가입() {
        // given
        
        // when
        
        // then
    }

 

테스트를 할때는 //given, // when, // then 이렇게 세부분으로 나눠서 작성하면 이후에 확인하기 좋다고!

 

 

일단 정상플로우를 테스트해본다.

@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");

 

이렇게 spring 같은 값을 넣는다고 하면 에러가 발생한다 ( 우리는 같은 이름의 회원은 받지않기로 앞서 service단계에서 정했으니까! )

그럼 try-catch 방식으로 한번 예외처리를 해보자

 

 

우선 member1의 객체를 가져오고

// when
memberService.join(member1);
  try {
      memberService.join(member2);
     fail();
   }catch (IllegalStateException e){
      assertThat(e.getMessage()).isEqualTo("이미 존재하는 회원입니다.");
      //Assertions를 junit으로 사용해야 asserThat을 쓸 수 있다.
   }

 

이렇게 코드를 작성하면 중복을 잡아줘서 테스트도 성공이다.

근데 코드가 너무기니까 좀더 단순하게 바꿔줄 수 있다.

assertThrows(IllegalStateException.class, () -> memberService.join(member2));

 

Assertions의 assertThrows를 통해 예외처리를 한줄에 해줄 수 있다. 

변수 추출하기를 단축키이용해서 (Ctrl + Alt + v

 

위의 값을 변수에 넣어주고 중복메시지가 제대로 나오는지 확인! 

IllegalStateException e = assertThrows(IllegalStateException.class, () -> memberService.join(member2));
assertThat(e.getMessage()).isEqualTo("이미 존재하는 회원입니다.");

 

전체코드를 테스트할때 각 메서드 clear 해주는 작업을 이번에도 해주자

 

저번에 repository단 테스트 할때처럼 

MemoryMemberRepository repository = new MemoryMemberRepository();

@AfterEach
public void afterEach(){
    repository.clearStore();
}

 

이렇게 코드 추가하면 되는거아닌가? 싶다. 이렇게도 가능한데 여기서 생성된  MemoryMemberRepository랑 

service단에서 생성된

private final MemberRepository memberRepository = new MemoryMemberRepository();

이 객체랑 인스턴스가 달라서 나중에 문제가 생길 수 있기때문에 각각 new로 생성해주지 않고 같은것을 사용하도록 하는것이 좋다

 

그렇다면!

일단 MemoryService에서 아래 코드처럼 수정해준다.

private final MemberRepository memberRepository;

//alt+insert
public MemberService(MemberRepository memberRepository) {
    this.memberRepository = memberRepository;
} 

 이때 Alt+insert 단축기를 사용해서 생성자를 외부에서 가져올 수 있도록 해주자

 

그리고 test단에서

MemberService memberService;
MemoryMemberRepository memberRepository;

@BeforeEach
public void beforeEach(){
    memberRepository = new MemoryMemberRepository();
    memberService = new MemberService(memberRepository);
}

@BeforeEach를 넣어서 실행되기전에 객체를 생성할 수 있게해준다 

 

 

사실 완벽하게 이해가 가진 않지만 이게 Dependency Injection, DI 라고 한다 (의존성 주입)

다음시간에 DI에 대해서 자세히 배운다고하니 그때 좀더 이해해봐야겠다 ! ! !