회원 관리 예제 - 벡엔드 개발 (1) : 비즈니스 요구사항 정리

sylvie·2021년 11월 16일
0

스프링부트

목록 보기
8/11
post-custom-banner

비즈니스 요구사항 정리

  • 데이터: 회원ID, 이름
  • 기능: 회원 등록, 조회
  • 아직 데이터 저장소가 선정되지 않음(가상의 시나리오)

일반적인 웹 애플리케이션 계층 구조

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

클래스 의존관계

  • 아직 데이터 저장소가 선정되지 않아서, 우선 인터페이스로 구현 클래스를 변경할 수 있도록 설계
  • 데이터 저장소는 RDB, NoSQL 등등 다양한 저장소를 고민중인 상황으로 가정
  • 개발을 진행하기 위해서 초기 개발 단계에서는 구현체로 가벼운 메모리 기반의 데이터 저장소 사용

회원 도메인과 리포지토리 만들기

회원 객체

  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.*;
  /**
* 동시성 문제가 고려되어 있지 않음, 실무에서는 ConcurrentHashMap, AtomicLong 사용 고려
*/
  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 List<Member> findAll() {
 		return new ArrayList<>(store.values());
	  }
      
      @Override
      public Optional<Member> findByName(String name) {
          return store.values().stream()
                  .filter(member -> member.getName().equals(name))
                  .findAny();
      }
      
      public void clearStore() {
          store.clear();
      } 
   }

회원 리포지토리 테스트 케이스 작성

개발한 기능을 실행해서 테스트 할 때 자바의 main 메서드를 통해서 실행하거나, 웹 애플리케이션의 컨트롤러를 통해서 해당 기능을 실행한다. 이러한 방법은 준비하고 실행하는데 오래 걸리고, 반복 실행하기 어렵고 여러 테스트를 한번에 실행하기 어렵다는 단점이 있다. 자바는 JUnit이라는 프레임워크로 테스트를 실행해서 이러한 문제를 해결한다. 또한 클래스 전체를 실행해서 전체 테스트를 실행할 수 있다는 장점이 있다.

회원 리포지토리 메모리 구현체 테스트

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 java.util.Optional;
  import static org.assertj.core.api.Assertions.*;

class MemoryMemberRepositoryTest {
   MemoryMemberRepository repository = new MemoryMemberRepository();
    @AfterEach
    public void afterEach() {
        repository.clearStore();
    }
    @Test
    public void save() {
//given
        Member member = new Member();
        member.setName("spring");
//when
        repository.save(member);
//then
        Member result = repository.findById(member.getId()).get();
        assertThat(result).isEqualTo(member);
    }
    @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);
      }
}

save()를 run했을때, 같다면 message가 console에 찍히진않지만

이렇게 초록색 ✔️가 표시된다.
다르다면 아래와 같이

console에
'expected: hello.hellospring.domain.Member@7922d892 but was:
Expected :hello.hellospring.domain.Member@7922d892
Actual :null' 라고 에러 메세지가 찍히고
초록색 ✔️가 아닌 ☓가 표시되는 것을 볼 수 있다.

또한 파일을 run하면 전체 test를 한번에 실행 할 수 있다는 장점이 있다.

하지만, 이번에 전체 class를 run했을 때 이전에 정상 실행되었던 findByName()이 에러가 나는 것을 발견할 수 있다.

이유는 test클래스는 실행순서는 보장하지않는다.
위 사진을 보면 findAll()이 findByName()보다 먼저 실행된 것을 볼 수 있다. 이 때문에 객체에 저장된값이 달라 에러가 난것이다.
이런 경우를 방지하기 위해선, 항상 객체를 clear해줘야한다.

post-custom-banner

0개의 댓글