Framework & Library/springboot
[Spring] #3 회원관리 예제 - 백엔드 개발
hi._.0seon
2021. 1. 21. 15:42
반응형
1. 비즈니스 요구사항 정리
- 데이터: 회원ID, 이름
- 기능: 회원 등록, 조회
- 아직 데이터 저장소가 선정되지 않음(가상의 시나리오)
컨트롤러: 웹 MVC의 컨트롤러 역할
서비스: 핵심 비즈니스 로직 구현
리포지토리: 데이터베이스에 접근, 도메인 객체를 DB에 저장하고 관리
도메인: 비즈니스 도메인 객체, 예) 회원, 주문, 쿠폰 등등 주로 데이터베이스에 저장하고 관리됨
- 아직 데이터 저장소가 선정되지 않아서, 우선 인터페이스로 구현 클래스를 변경할 수 있도록 설계
- 데이터 저장소는 RDB, NoSQL 등등 다양한 저장소를 고민중인 상황으로 가정
- 개발을 진행하기 위해서 초기 개발 단계에서는 구현체로 가벼운 메모리 기반의 데이터 저장소 사용
2. 회원 도메인과 레포지토리 만들기
Member.java
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;
}
}
- id: 회원 id 가 아니라 시스템이 데이터를 구분하기 위해 사용하는 id
MemoryMemberRepository.java
public class MemoryMemberRepository implements MemberRepository{
private static Map<Long, Member> store = new HashMap<>();
// 실무에서는 동시성 문제가 발생할 수 있어 ConcurrentHashMap 사용
private static long sequence = 0L;
//0, 1, 2 키 값 생성해줌 (동시성 문제로 atomicLong 사용)
@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));
// NULL 이 반환될 가능성이 있는 경우 옵셔널로 감싸서 반환 -> 클라이언트에서 처리
}
@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());
}
}
3. 회원 레포지토리 테스트 케이스 작성
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();
// 옵셔에서 값을 꺼낼 때는 .get() 으로 꺼낼 수 있다
// System.out.println("result = " + (result == member));
// Assertions.assertEquals(result, member); // expected, actual
// Assertions.assertThat(member).isEqualTo(result);
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);
}
}
- Assertions : org.assertj.core.api
좀더 편하게 쓸 수 있는 애
- 클래스 수준에서 실행시키면 클래스에 포함된 테스트케이스 전부 실행됨
- 테스트가 실행되는 순서는 보장이 안됨
- 모든 테스트는 순서와 상관없이 실행되도록 해야 함
@AfterEach
- callBack 메서드
- 메서드가 끝날 때마다 호출되는 메서드
- 한 테스트가 끝나면 공용 변수에 Member가 저장되기 때문에 그 것을 전부 지운 후 다음 테스트 실행하도록 작성
# TDD
- 테스트케이스를 작성한 후 코드를 작성하는 것
- 테스트 주도 개발
4. 회원 서비스 개발
테스트 코드는 대상 클래스의 패키지 경로와 같은 곳에 생성
cmd + shift + T 로 생성 가능
assertThat
- 코드를 통과하면 테스트 성공, 예상 결과와 다른 경우 테스트 실패로 뜬다
jongmin92.github.io/2020/03/31/Java/use-assertthat/
테스트 방법
- // given
주어진 상황 - // when
실행했을 때 - // then
이러한 결과가 나와야 한다
반응형