가볍게 회원가입, 전체회원 조회 비지니스 로직을 구현했다.
말도안되지만 연습이니까 만약 이름이 같다면 회원가입을 못하게 막았다.(나중엔 아이디로 해야한다)
여기서 private final MemberRepository memberRepository = new MemoryMemberRepository();
이와 같이 사용하면 두 클래스간에 결합도가 강해지기 때문에 생성자를 이용해 의존성 주입(DI)를 했다.
@Autowired
어노테이션을 이용하면 IoC컨테이너 안에 있는 빈을 자동으로 주입해준다.
package hello.hellospring.service;
import hello.hellospring.domain.Member;
import hello.hellospring.repository.MemberRepository;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import java.util.List;
import java.util.Optional;
@Service
public class MemberService {
private final MemberRepository memberRepository;
@Autowired
public MemberService(MemberRepository memberRepository) {
this.memberRepository = memberRepository;
}
/*/*
* 회원가입
*/
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);
}
}
로그인시 정상적으로 작동할때 테스트 코드와 같은 이름으로 가입했을때 에러메세지를 띠우는 테스트 코드를 작성했다.
@BeforeEach
는 각각의 테스트 실행 전에 호출되는데. 테스트가 서로 영향이 없도록 항상 새로운 객체를 생성하고, 의존관계도 새로 맺어준다.
package hello.hellospring.service;
import hello.hellospring.domain.Member;
import hello.hellospring.repository.MemoryMemberRepository;
import org.assertj.core.api.Assertions;
import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
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 join() {
//given 무언가가 주어졌는데
Member member = new Member();
member.setName("spring");
//when 이게 주어졌을떄
Long saveId = memberService.join(member);
//then 그러면이게 나와야돼
Member findMember = memberService.findOne(saveId).get();
Assertions.assertThat(member.getName()).isEqualTo(findMember.getName());
}
@Test
public void validateDuplicateMember() {
//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));
Assertions.assertThat(e.getMessage()).isEqualTo("이미 존재하는 회원입니다");
}
@Test
void findMembers() {
}
@Test
void findOne() {
}
}
스프링에서 빈에 등록하는 방법은 두가지가 있다.
1. 컴포넌트 스캔과 자동 의존관계 설정
2. 자바 코드로 직접 스프링 빈 등록하기
여기서 컴포넌트는 싱글톤 클래스 빈을 생성하는 어노테이션이다.
@Component
를 정의하면 스프링에다가 빈으로 등록할 수 있다.
우리는 이미 @Coponent
를 썼다. @Controller
,@Service
,@Repository
에서 말이다.
아래와 같이 @Service
를 까보면 @Component
를 포함하고 있는걸 볼 수 있다.
이렇게 컴포넌트를 정의하면 @SpringBootApplication
에서 컴포넌트를 스캔해서 의존관계가 맺어진다.
생성자에 @Autowired
가 있으면 스프링이 연관된 객체를 스프링 컨테이너에서 찾아서 넣어준다. 이렇게 객체 의존관계를 외부에서 넣어주는 것을 DI (Dependency Injection), 의존성 주입
이라 한다.
@Service
public class MemberService {
private final MemberRepository memberRepository;
@Autowired
public MemberService(MemberRepository memberRepository) {
this.memberRepository = memberRepository;
}
참고로 스프링은 컨테이너에 빈을 등록할때 싱글톤 패턴으로 등록하기 때문에 같은 스프링 빈이면 모두 같은 인스턴스
이다. (설정으로 싱글톤 패턴을 안사용할 수도 있다.)