[Spring 복습] Spring 입문 - 회원관리 예제

seonjeong·2023년 11월 17일
0

Spring

목록 보기
21/27
post-thumbnail

❤️ 비즈니스 요구사항 정리

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

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

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

클래스 의존관계

  • 아직 데이터 저장소가 선정되지 않아, 인터페이스로 구현 클래스를 변경할 수 있도록 설계
  • 초기 개발 단계에서는 구현체로 가벼운 메모리 기반의 데이터 저장소 사용

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

회원 객체

src/main/java/hello.hello-spring 하위에 domain 패키지를 새로 만들고 Member.java파일 생성 후 다음과 같이 작성

package hello.hellospring.domain;

public class Member {
    private Long id;	  // 회원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;
    }
}

회원 리포지토리 인터페이스 작성

src/main/java/hello.hello-spring 하위에 repository 패키지를 새로 만들고 MemberRepository.java 인터페이스 파일 생성 후 다음과 같이 작성

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();  // 회원 목록
}
  • findById, findByName 메서드에서 NullPointException이 발생할 수 있기 때문에 Optional 사용

    Optional

    : null이 올 수 있는 값을 감싸는 Wrapper 클래스

    • of() : 값이 절대 null이 아닌 경우에 사용
    • ofNullable() : 값이 null일수도 있고 아닐수도 있는 경우에 사용

회원 리포지토리 메모리 구현체 작성

/repository 패키지 하위에 MemoryMemberRopository.java 파일 생성 후 다음과 같이 작성

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));  //  null이면 Optional.empty반환
    }

    @Override
    public Optional<Member> findByName(String name) {
        return store.values().stream()
                .filter(member -> member.getName().equals(name))
                .findAny();  // name과 일치하는 회원을 찾아 반환
    }

    @Override
    public List<Member> findAll() {
        return new ArrayList<>(store.values());
    }
    
    public void clearStore() {
        store.clear();
    }
}
  • 동시성 문제는 고려되어 있지 않음
  • 실무에서는 ConcurrentHashMap<>AtomicLong 사용 고려

❤️ 회원 서비스 개발

src/main/java/hello.hello-spring 하위에 service 패키지를 새로 만들고 MemberService.java파일 생성 후 다음과 같이 작성

package hello.hellospring.service;

import hello.hellospring.domain.Member;
import hello.hellospring.repository.MemberRepository;
import hello.hellospring.repository.MemoryMemberRepository;

import java.util.List;
import java.util.Optional;

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

    /**
     * 회원 가입
     */
    public Long join(Member member) {
        // 같은 이름이 있는 중복 회원 X
        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);
    }
}
  • 서비스 클래스에서는 비즈니스 용어를 사용해야 함

의존성 주입(DI)

DI(Dependency Injection)

  • 객체가 의존하는 또 다른 객체를 외부에서 선언하고 이를 주입받아 사용하는 것
  • 의존성을 주입하는 방법으로는 생성자 주입, setter 주입, 필드 주입이 있다
  • 의존성을 주입하면 모듈간 의존성이 줄어들고, 재사용성이 높아진다

참고 - 의존관계 주입(Dependency Injection) 쉽게 이해하기

기존 코드는 회원 서비스가 메모리 회원 리포지토리를 직접 생성하게 함

private final MemberRepository memberRepository = new MemoryMemberRepository();

다음과 같이 하면, MemberService 객체를 생성할 때 외부에서 memberRepository 객체를 주입받아 사용할 수 있게 됨

// 생성자 주입
private final MemberRepository memberRepository;

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

❤️ 테스트 케이스 작성

  • 개발한 기능이 정상적으로 작동하는지 테스트 할 때 테스트 케이스를 작성하여 검증
  • JUnit이라는 프레임워크로 테스트
  • IntelliJ Alt + Enter(윈도우) 단축키로 생성 가능

테스트 코드

: 작성한 코드가 의도한 대로 구동되는지 확인하는 코드

  • given - when - then 패턴으로 작성할 수 있음 ( → 준비-실행-검증)
    • given : 테스트 하기 위해 기본적으로 세팅하는 값
    • when : 테스트를 하기 위한 조건을 지정
    • then : 테스트 하기 위한 행위가 우리가 예상하는대로 동작하는지 검증하는 행동/절차

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

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

src/test/java/hello.hello-spring 하위에 repository 패키지를 만들고 MemoryMemberRepositoryTest.java 파일 생성 후 다음과 같이 작성

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;

import static org.assertj.core.api.Assertions.*;

public 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();  
        // 검증 → 저장한 회원객체와 DB의 회원객체가 일치여부 확인
        // Assertions.assertEquals(member, result);
        assertThat(member).isEqualTo(result);
    }

    @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);
    }
};
  • 테스트 순서는 보장되지 않기 때문에 모든 테스트는 순서와 상관없이 메서드별로 따로 동작하게 설계해야 함 (독립적으로 설계할 것)

  • 한 번에 여러 테스트를 실행하면 메모리 DB에 직전 테스트의 결과가 남을 수 있음

  • 이렇게 되면 이전 테스트로 인해 다음 테스트가 실패할 가능성이 있음
    → 테스트가 하나 끝나면 데이터를 클리어 해주어야 함 : @AfterEach 사용

  • 테스트 케이스를 먼저 작성 후 개발 가능 → TDD(테스트 주도 개발)

  • 테스트 결과 - 검증 완료

회원 서비스 테스트 케이스 작성

package hello.hellospring.service;

import hello.hellospring.domain.Member;
import hello.hellospring.repository.MemoryMemberRepository;
import org.junit.jupiter.api.AfterEach;
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
    void 회원가입() {
        // given : 회원 정보가 주어짐
        Member member = new Member();
        member.setName("judyan");

        // when : 회원가입 실행
        Long saveId = memberService.join(member);

        // then : 가입여부 검증
        Member findMember = memberRepository.findById(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");

        // when
        memberService.join(member1);
        IllegalStateException e = assertThrows(IllegalStateException.class, () -> memberService.join(member2));  // 예외가 발생해야 함

        assertThat(e.getMessage()).isEqualTo("이미 존재하는 회원입니다.");
/*
        try {
            memberService.join(member2);
            fail();
        }
        catch (IllegalStateException e) {
            assertThat(e.getMessage()).isEqualTo("이미 존재하는 회원입니다.");
        }
*/
    }
}
  • @BeforeEach : 각 테스트 실행 전에 호출된다. 테스트가 서로 영향이 없도록 항상 새로운 객체를 생성하고, 의존관계도 새로 맺어줌
  • 중복_회원_예외() 메서드는 동일한 이름의 회원을 회원가입시켜 예외가 발생하는지 확인하는 테스트이며 try-catch문을 사용하여 작성하는 방법도 있음

IntelliJ 단축키

  • Getter, Setter, Constructor 등 생성 : Alt + Insert
  • 선택 영역에서 치환 : Shift + Alt + R
  • Refactor this : Shift + Ctrl + Alt + T
    • Extract Method : 메서드 추출 리팩토링
  • Create Test : Alt + Enter
  • 이전실행을 재실행 : Shift + F10



출처 - 인프런 '스프링 입문' 강의

profile
🦋개발 공부 기록🦋

0개의 댓글

관련 채용 정보