아직 데이터 저장소를 선정하지 않은 상황으로 가정 - RDB, NoSQL, JPA 중 고민하는 상황
📂 src/main/java/.../domain/Member
@Entity
public class Member {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
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/.../repository/MemberRepository
null
로 반환될 수 있는 값 -> Optional
로 감싸서 반환public interface MemberRepository {
Member save(Member member); // 저장
Optional<Member> findById(Long id); // id로 회원 조회
Optional<Member> findByName(String name); // name으로 회원 조회
List<Member> findAll(); // 모든 회원 조회
}
📂 src/main/java/.../repository/MemoryMemberRepository
public class MemoryMemberRepository implements MemberRepository {
/* 📌 공유변수 store, sequence
* 동시성 문제를 고려해야 함(예제에서는 동시성 문제 고려❌)
* 실무에서는 ConcurrentHashMap, AtomicLong 사용 고려
*/
private static Map<Long, Member> store = new HashMap<>( );
private static long sequence = 0L; // 0, 1, 2 키값을 자동으로 생성하는 역할
@Override
public Member save(Member member) {
member.setId(++sequence);
store.put(member.getId(), member);
return member;
}
@Override
public Optional<Member> findById(Long id) {
// Map<Key, Value> 형태로 저장, Map의 get(key)는 value를 반환
// ofNullable(value)는 value가 null일 수도 있고, 아닐 수도 있을 때 사용
return Optional.ofNullable(store.get(id));
}
@Override
public Optional<Member> findByName(String name) {
// values()는 Map에 저장된 모든 value객체(Member)를 Collection으로 반환
return store.values().stream()
.filter(member -> member.getName().equals(name))
// 필터링된 요소 중 조건을 만족하는 임의의 요소를 Optional로 반환
.findAny();
}
@Override
public List<Member> findAll() {
return new ArrayList<>(store.values());
}
public void clearStore(){
store.clear();
}
}
.steam()
연산을 할 수 있는 대표적인 자료형.steam()
연산을 사용할 수 있는 자료형들은Collection
인터페이스(List
, Set
, Queue
)Map
인터페이스(keySet()
, entrySet()
)Arrays
)main
메서드를 통해서 실행하거나,JUnit
이라는 프레임워크로 테스트를 실행해서 이러한 문제를 해결📂MemoryMemberRepositoryTest.java
class MemoryMemberRepositoryTest {
MemberRepository repository = new MemoryMemberRepository();
// 📍 findById() 테스트
@Test
public void findByIdTest(){
// given
Member member = new Member();
member.setName("나는의연");
// when
repository.save(member);
// then - 검증
// findById()의 반환타입은 Optional -> Member와 타입 불일치 해소를 위해 get() 사용
// get()은 Optional 내부에 저장된 값을 꺼냄(Optional을 까는 메서드)
Member result = repository.findById(member.getId()).get();
// 검증 방법1 - assertThat() : JUnit, AssertJ 라이브러리의 테스트 검증 메서드
assertThat(result).isEqualTo(member);
// 검증 방법2 - assertEquals(expected(기대값), actual(실제값))
assertEquals(member, result);
// 검증 방법3
System.out.println("result = " + (result == member)); // ✅ true
}
// 📍 findByName() 테스트
@Test
public void findByNameTest(){
Member member1 = new Member();
member1.setName("나는최강의연");
repository.save(member1);
Member member2 = new Member();
member2.setName("나는지존의연");
repository.save(member2);
Member result = repository.findByName(member1.getName()).get();
assertThat(result).isEqualTo(member1);
// assertThat(result).isEqualTo(member2); // ❌ test failed
// assertAll(
// () -> assertEquals(result, member1),
// () -> assertEquals(result, member2)
// );
}
// 📍 findAll() 테스트
@Test
public void findAllTest(){
Member member1 = new Member();
member1.setName("zizon의연v");
repository.save(member1);
Member member2 = new Member();
member2.setName("zizon지연v");
repository.save(member2);
List<Member> result = repository.findAll();
assertThat(result.size()).isEqualTo(2);
assertEquals(result.size(), 2);
}
}
📂MemoryMemberRepository.java
public void clearStore(){
store.clear();
}
@AfterEach
public void afterEach(){
repository.clearStore();
}
class MemoryMemberRepositoryTest {
MemoryMemberRepository repository = new MemoryMemberRepository();
@AfterEach
public void afterEach(){
repository.clearStore();
}
// 📍 findById() 테스트
@Test
public void findByIdTest(){
// given
Member member = new Member();
member.setName("나는의연");
// when
repository.save(member);
// then
Member result = repository.findById(member.getId()).get();
assertThat(result).isEqualTo(member);
assertEquals(member, result);
System.out.println("result = " + (result == member)); // ✅ true
}
// 📍 findByName() 테스트
@Test
public void findByNameTest(){
Member member1 = new Member();
member1.setName("나는최강의연");
repository.save(member1);
Member member2 = new Member();
member2.setName("나는지존의연");
repository.save(member2);
Member result = repository.findByName(member1.getName()).get();
assertThat(result).isEqualTo(member1);
}
// 📍 findAll() 테스트
@Test
public void findAllTest(){
Member member1 = new Member();
member1.setName("zizon의연v");
repository.save(member1);
Member member2 = new Member();
member2.setName("zizon지연v");
repository.save(member2);
List<Member> result = repository.findAll();
assertThat(result.size()).isEqualTo(2);
assertEquals(result.size(), 2);
}
}
@BeforeEach
는 각 테스트 메서드 실행 전에 필요한 설정 작업을 수행할 때 사용save()
, findById()
, findByName()
, findAll()
join()
, validDuplicatedMember()
@Service
@Transactional
public class MemberService {
private final MemberRepository memberRepository;
public MemberService(MemberRepository memberRepository){
this.memberRepository = memberRepository;
}
// 회원가입 - 동명이인 회원 중복 가입❌
public Long join(Member member){
validateDuplicatedMember(member);
memberRepository.save(member);
return member.getId();
}
// 중복 회원 확인 -> 중복 회원가입시 exception이 터지는지 검증이 필요(테스트 케이스 활용)
private void validateDuplicatedMember(Member member) {
memberRepository.findByName(member.getName())
.ifPresent(m -> {
throw new IllegalStateException("이미 존재하는 회원입니다.");
});
}
// 전체 회원 조회
public List<Member> findMembers(){
return memberRepository.findAll();
}
// 단일 회원 조회
public Optional<Member> findMember(Long memberId){
return memberRepository.findById(memberId);
}
}
ifPresent(Cosumer<? super T> action)
:🍀 테스트 케이스는
given - when - then
구조로 작성할 것
given
: 주어진 상황, 조건when
: 실행되었을 때then
: 기대되는 결과class MemberServiceTest {
MemberService memberService;
MemoryMemberRepository memberRepository;
// Test에서 사용하는 레포지토리와 서비스의 레포지토리를 일치시키기 위함
// Test와 서비스에서 각각 new로 레포지토리를 생성할 경우 애매~함(굳이 두 개의 객체를 사용할 이유가 없음)
@BeforeEach
void beforeEach(){
memberRepository = new MemoryMemberRepository();
memberService = new MemberService(memberRepository);
}
@AfterEach
void afterEach(){
memberRepository.clearStore();
}
@Test
void 회원가입() {
// given
Member member = new Member();
member.setName("zizon의연v");
// when
Long savedId = memberService.join(member);
// then
// assertEquals(2, savedId);
// Service의 findMember()로 테스트
Member findMember1 = memberService.findMember(savedId).get();
assertThat(member.getName()).isEqualTo(findMember1.getName());
// Repository의 findById()로 테스트
Member findMember2 = memberRepository.findById(savedId).get();
}
// 테스트는 정상 플로우보다 예외 플로우 검증이 훨씬 중요
@Test
void 중복_회원_예외(){
// given
Member member1 = new Member();
member1.setName("zizon의연v");
Member member2 = new Member();
member2.setName("zizon의연v");
// when
memberService.join(member1);
// then
// try {
// memberService.join(member2);
// fail("예외가 발생해야합니다.");
// } catch (IllegalStateException e){
// assertThat(ex.getMessage()).isEqualTo("이미 존재하는 회원입니다.");
// }
// ♻️ try-catch문 리팩토링
IllegalStateException e = assertThrows(IllegalStateException.class, () -> memberService.join(member2));
assertThat(e.getMessage()).isEqualTo("이미 존재하는 회원입니다.");
}
}
MemberService
의 레포지토리와 MemberServiceTest
의 레포지토리가DI
하도록 코드 작성📂MemberService.java
@Service
@Transactional
public class MemberService {
private final MemberRepository memberRepository;
// DI, Dependency Injection
public MemberService(MemberRepository memberRepository){
this.memberRepository = memberRepository;
}
...
}
org.junit.jupiter.api.Assertions.fail()
(JUnit 5 기준)