상세 컨텐츠

본문 제목

[스프링 1팀] 스프링 입문 섹션4

24-25/Spring 1

by oze 2024. 10. 4. 10:00

본문

728x90

[ 회원 관리 예제 ]

비지니스 요구사항 정리

  • 데이터: 회원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();
    }
 }

 


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

자바는 JUnit이라는 프레임워크로 테스트를 실행

 

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

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);
    }
 }
  • @AfterEach : 한번에 여러 테스트를 실행하면 메모리 DB에 직전 테스트의 결과가 남을 수 있어  다음 테스트가 실패할 가능성이 있다. @AfterEach를 사용하면 각 테스트가 해당 기능을 실행한다. 여기서는 메모리 DB에 저장된 데이터를 삭제한다.
  • 테스트는 각각 독립적으로 실행되어야 한다. 테스트 순서에 의존관계가 있는 것은 좋은 테스트가 아니다.

회원 서비스 개발

 package hello.hellospring.service;

 import hello.hellospring.domain.Member;
 import hello.hellospring.repository.MemberRepository;
 
 import java.util.List;
 import java.util.Optional;
 
 public class MemberService {
 	private final MemberRepository memberRepository = new MemoryMemberRepository();
    
    /**
     *회원가입
     */
 	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);
    }
 }

 


회원 서비스 테스트

기존에는 회원 서비스가 메모리 회원 리포지토리를 직접 생성

 public class MemberService {
    private final MemberRepository memberRepository = new MemoryMemberRepository();
 }

 

회원 서비스 코드를 DI(의존성 주입) 가능하게 변경

public class MemberService {

 	private final MemberRepository memberRepository;
 	
    public MemberService(MemberRepository memberRepository) {
 		this.memberRepository = memberRepository;
    }
    ...
 }

 

회원 서비스 테스트

package hello.hellospring.service;
 
 import hello.hellospring.domain.Member;
 import hello.hellospring.repository.MemoryMemberRepository;
 import org.junit.jupiter.api.BeforeEach;
 import org.junit.jupiter.api.Test;
 
 import static org.assertj.core.api.Assertions.*;
 import static org.junit.jupiter.api.Assertions.*;
 
 class MemberServiceTest {
 	MemberService memberService;
 	MemoryMemberRepository memberRepository;
    
    @BeforeEach
 	public void beforeEach() {
        memberRepository = new MemoryMemberRepository();
        memberService = new MemberService(memberRepository);
    }
    
    @AfterEach
 	public void afterEach() {
        memberRepository.clearStore();
    }
    
    @Test
 	public void 회원가입() throws Exception {
        //Given
        Member member = new Member();
        member.setName("hello");

        //When
        Long saveId = memberService.join(member);

        //Then
        Member findMember = memberRepository.findById(saveId).get();
        assertEquals(member.getName(), findMember.getName());
    }
    
    @Test
 	public void 중복_회원_예외() throws Exception {
        //Given
        Member member1 = new Member();
        member1.setName("spring");
        Member member2 = new Member();
        member2.setName("spring");
        //When
        memberService.join(member1);
         IllegalStateException e = assertThrows(IllegalStateException.class,
                    () -> memberService.join(member2));//예외가 발생해야 한다.

        assertThat(e.getMessage()).isEqualTo("이미 존재하는 회원입니다.");
    }
 }
  • @BeforeEach : 각 테스트 실행 전에 호출된다. 테스트가 서로 영향이 없도록 항상 새로운 객체를 생성하 고, 의존관계도 새로 맺어준다.

QUIZ

  1. 일반적인 웹 애플리케이션 계층 구조에서 _____가 데이터베이스에 접근, 도메인 객체를 DB에 저장하고 관리한다.
  2. 일반적인 웹 애플리케이션 계층 구조에서 핵심 비즈니스 로직을 구현하는 계층을 ___라고 한다.
  3. 일반적인 웹 애플리케이션 계층 구조에서 주로 데이터베이스에 저장하고 관리되는 것을 ___이라고 한다.
  4. 자바에서 테스트를 실행하는 프레임워크에는 _____가 있다.
  5. __________을(를) 사용하면 각 테스트가 종료될 때 호출된다.
  6. ‘외부'에서 클라이언트에게 서비스를 제공(주입)하는 것을 __(___ __)이라고 한다.
  7. ___________을(를) 사용하면 각 테스트 실행 전에 호출된다.

PROGRAMMING QUIZ

1. 해당 리포지토리 구현체의 save 기능에 대해 테스트하는 코드를 작성하시오.

// (...)
 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;
    }
    
    //(...)
 }

 

2. 아래 코드의 회원 가입 기능에 대해 이미 가입된 이름으로 회원가입을 하려는 중복 예외 경우에 대한 테스트 코드를 작성하시오.

public class MemberService {
 	private final MemberRepository memberRepository = new MemoryMemberRepository();
   
    /**
     * 회원가입
     */
 	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("이미 존재하는 회원입니다.");
                });
    }
	
    //(...)
}

  1. 리포지토리
  2. 서비스
  3. 도메인
  4. JUnit
  5. @AfterEach
  6. DI(의존성 주입)
  7. @BeforeEach

1번

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

 

2번

 @Test
 public void 중복_회원_예외() throws Exception {
 	//Given
 	Member member1 = new Member();
	member1.setName("spring");
 
 	Member member2 = new Member();
	member2.setName("spring");
 
 	//When
    memberService.join(member1);
 	IllegalStateException e = assertThrows(IllegalStateException.class,
                () -> memberService.join(member2));//예외가 발생해야 한다.
                
 	assertThat(e.getMessage()).isEqualTo("이미 존재하는 회원입니다.");
}

 


출처:  스프링 입문, 코드로 배우는 스프링 부트, 웹MVC, DB접근기술 강의

Corner Spring 1
Editor:  Minyong

728x90

관련글 더보기