[스프링 입문] 정리노트 - 3
스프링 입문 - 코드로 배우는 스프링 부트, 웹 MVC, DB 접근 기술 강의
시작
회원 관리 서비스 만들기
이제 기본적인 지식을 얼추 쌓았습니다. 실제로 서비스를 만들어보도록 하겠습니다. 다음과 같은 순서로 진행합니다.
- 비즈니스 요구사항 정리
- 회원 도메인과 리퍼지토리 만들기
- 회원 리퍼지토리 테스트 케이스 작성
- 회원 서비스 개발
- 회원 서비스 테스트
비즈니스 요구사항 정리
데이터베이스 및 제공 기능은 다음과 같습니다.
- 데이터 : 회원 ID, 이름
- 기능 : 회원 등록, 조회
계층 구조는 다음과 같습니다.
- 컨트롤러 : 웹 MVC의 컨트롤
- 서비스 : 핵심 비즈니스 로직 구현
- 리포지토리 : 데이터베이스에 접근, 도메인 객체를 DB에 저장하고 관리
- 도메인 : 비즈니스 도메인 객체 ex)회원, 주문, 쿠폰 등 주로 DB에 저장하고 관리됨.
또한 가상의 시나리오를 추상화하기 위해 memberService
는 MemberRepository
라는 인터페이스
를 구현하게됩니다. 이유는 실제 DB가 붙었을 때 유연하게 교체하기 위함입니다.
그 이전까지 데이터는 메모리에서 관리하는 방향으로 진행합니다.
회원 도메인과 리포지토리 만들기
src/main/java/hello/hello/spring/domain
경로에 Member
클래스를 생성합니다.
java
// Member.class
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;
}
}
그리고 리포지토리를 만들어봅시다.
src/main/java/hello/hello/spring/domain
경로에 MemberRepository.interface
를 생성합니다.
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은 findById가 없으면 null을 반환하도록 해줌.
Optional<Member> findById(Long id);
Optional<Member> findByName(String name);
List<Member> findAll();
}
이제 MemberRepository.interface
를 구현하는 구현체를 만들어봅시다.
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) {
// id 값을 부여
member.setId(++sequence);
store.put(member.getId(), member);
return member;
}
@Override
public Optional<Member> findById(Long id) {
// null 값도 허용
return Optional.ofNullable(store.get(id));
}
@Override
public Optional<Member> findByName(String name) {
// stream으로 store내 member객체를 순회하여
// 이름이 겹치는 조건이 있는지 검색한다.
return store.values().stream()
.filter(member -> member.getName().equals(name))
.findAny();
}
@Override
public List<Member> findAll() {
// member들을 List로 넘겨준다.
return new ArrayList<>(store.values());
}
}
이제 기능이 정상적으로 작동하는지 테스트케이스를 작성해봅시다.