비지니스 요구사항은 다음과 같다.
데이터: 회원 ID, 이름
기능: 회원등록, 조회
DB: 아직 선정되지 않음(가상의 시나리오)
웹 어플리케이션의 구조는 다음과 같다.
클래스 의존관계는 다음과 같은데 DB가 선정되지 않았기에 인터페이스로 구현 클래스를 변경할 수 있도록 설계하였다. 개발을 수행하기 위해 일단 메모리 기반의 DB를 사용할 것이다.
hello-spring디렉토리에 domain 패키지를 만들어 Member객체를 구현할 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와 이름, 기능으로는 회원 등록 및 조회 뿐이므로 위와 같은 구현체를 만든다.
다음 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);//null을 처리하기 위한 optional
Optional<Member> findByName(String name);
List<Member> findAll();
}
이제 해당 인터페이스를 메모리 구현체로 구현해보자.
repository 디렉토리에 MemoryMemberRepository.java를 만든다.
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; //sequence는 0, 1, 2처럼 key값을 생성
@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();
}
}
다음과 같이 메모리 위에 저장되는 레파지토리를 구현하였다. 다음에는 해당 레파지코리를 사용하는 멤버 서비스를 구현하도록 하자.
공감하며 읽었습니다. 좋은 글 감사드립니다.