📹 참고 : 인프런 [ 스프링 입문 - 김영한 ]
스프링부트를 이용하여 회원 관리 예제를 만들어 보자.
데이터: 회원ID, 이름
기능: 회원 등록, 조회
아직 데이터 저장소가 선정되지 않음(가상의 시나리오)


/java/hello.hellospring.domain/Member
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;
} }
java/hello/hellospring/repository/MemberRepository (인터페이스)
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은 null 관련
Optional<Member> findByName(String name);
List<Member> findAll();
}
java/hello/hellospring/repository/MemoryMemberRepository (구현체)
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)) //member의 getName이 파라미터로 넘어온 name 과 같은지 비교
.findAny(); //하나라도 찾으면 반 . 없으면 optional에 null이 포함돼서 반환
}
@Override
public List<Member> findAll() {
return new ArrayList<>(store.values()); //store의 Member 반환
}
}
JUnit 프레임워크를 활용하여 테스트 케이스를 작성해보자.
/src/test/java/hello/hellospring/java/hello/hellospring/repository/MemoryMemberRepositoryTest
파일명은 테스트하고자 하는 클래스 뒤에 Test를 붙이는 것이 관례.
또한 테스트케이스 클래스인 경우 굳이 public을 지정하지 않아도 된다.
package hello.hellospring.repository;
import hello.hellospring.domain.Member;
import org.assertj.core.api.Assertions;
import org.junit.jupiter.api.Test;
class MemoryMemberRepositoryTest {
MemberRepository repository = new MemoryMemberRepository();
@Test //어노테이션 필수
public void save(){
Member member = new Member();
member.setName("spring");
repository.save(member);
Member result=repository.findById(member.getId()).get(); //Optional에서 값을 꺼낼 때 get() 사용
Assertions.assertThat(member).isEqualTo(result);
//assertThat(테스트 타켓).메소드1().메소드2().메소드3();
// assertThat(actual).isEqualTo(expected): actual(실제값)이 expected(기대값)과 내용
}
}

save메서드의 테스트를 실행해 보면 초록색 체크와 함께 성공한 것을 알 수 있다.
@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();
Assertions.assertThat(result).isEqualTo(member1);
}
findByName 도 테스트를 해보면

제대로 실행 되는 것을 볼 수 있다.
이때
Assertions.assertThat(result).isEqualTo(member2);
member1 객체를 저장한 result와 member2를 비교하면

오류가 나는 것을 확인할 수 있다.
🌟테스트 케이스의 장점
클래스 레벨에서 돌리거나 전체 클러스를 동시에 다 테스트 해볼 수 있다.
findAll() 테스트
@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();
Assertions.assertThat(result.size()).isEqualTo(2);
//멤버 갯수 비교
}
Assertions.assertThat(result.size()).isEqualTo(3);
만약 3으로 비교한다면

기대값인 2와 다른 결과가 나온 것을 알 수 있다.
이번에는 3개의 메소드를 한번에 테스트를 해보자.

각각 테스트 할때는 오류가 없었는데 갑자기 오류가 생겼다.
=> findAll()이 먼저 실행되며 멤버 객체들이 혼합되었음.
모든 테스트는 순서랑 상관없이 메소드별로 다 따로 동작하게만 설계를 해야 함.
테스트가 하나 끝나고 나면 이 데이터를 클리어를 해줘야 함.
java/hello/hellospring/repository/MemoryMemberRepository에 아래 메소드를 추가해준다.
public void clearStore(){ store.clear(); //스토어를 비움}
/src/test/java/hello/hellospring/java/hello/hellospring/repository/MemoryMemberRepositoryTest 테스트 아래 코드를 추가해 준다.
@AfterEach //하나의 메서드가 끝날 때 마다 실행됨
public void afterEach(){
repository.clearStore();
}
그러면 테스트가 실행이 되고 끝날 때마다 한 번씩 저장소를 다 지운다.
package hello.hellospring.repository;
import hello.hellospring.domain.Member;
import org.assertj.core.api.Assertions;
import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.Test;
import java.util.List;
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(); //Optional에서 값을 꺼낼 때 get() 사용
Assertions.assertThat(member).isEqualTo(result);
//assertThat(테스트 타켓).메소드1().메소드2().메소드3();
// assertThat(actual).isEqualTo(expected): actual(실제값)이 expected(기대값)과 내용
}
@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();
Assertions.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();
Assertions.assertThat(result.size()).isEqualTo(2);
//멤버 갯수 비교
}
}
실행해 보면

순서와 상관 없이 전체 테스트가 진행 된 것을 알 수 있다.
🌟 TDD 란? Test Driven Development
선 개발 후 테스트 방식이 아닌 선 테스트 후 개발 방식의 프로그래밍 방법으로,
테스트를 먼저 만들고 구현 클래스를 만들어서 돌려보는 방법.
우리는 구현 클래스를 먼저 만들었기 때문에 TDD를 한 것은 아니다.