Member
와 MemoryMemberRepository
는 Java로 MemberRepository
는 Interface로 만들자.
Member
public class Member {
// Getter and Setter 만들기
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;
}
}
MemberRepository
public interface MemberRepository {
Member save(Member member);
Optional<Member> findByID(Long id); // Optional : 값이 없으면 null로 처리하는 방법
Optional<Member> findByName(String name);
List<Member> findAll();
}
MemoryMemberRepository
public class MemoryMemberRepository implements MemberRepository{
// 동시성 문제 고려 x
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이여도 사용 가능하게
}
@Override
public Optional<Member> findByName(String name) {
// 파라미터로 넘어온 name과 member에 있는 name이 같은 것을 필터링해서 반환
return store.values().stream()
.filter(member -> member.getName().equals(name))
.findAny();
}
@Override
public List<Member> findAll() {
return new ArrayList<>(store.values());
}
}
설명은 주석을 참고하자!
자바의 main 메서드를 통해서 실행하면 오래 걸리고, 반복 실행하기 어렵다는 단점이 있어서 JUnit이라는 프레임워크로 테스트를 실행한다.
src/text/java
하위 폴더에 생성한다.
public class MemoryMemberRepositoryTest {
MemoryMemberRepository repository = new MemoryMemberRepository();
@Test // save 테스트
public void save() {
Member member = new Member();
member.setName("spring"); // member를 저장하고
repository.save(member); // repository에 넣는다.
Member result = repository.findByID(member.getId()).get(); // Optional에서 get으로 값을 꺼낸다.
Assertions.assertThat(member).isEqualTo(result); // member와 result가 같은지 확인(Go에서 사용한거와 같음)
}
}
Run했을 때 성공하면 다음과 같은 결과가 나온다.
나머지 테스트 코드도 작성해보자.
...
@Test // findByName 테스트
public void findByName() {
Member member1 = new Member();
member1.setName("spring1");
repository.save(member1);
Member member2 = new Member();
member2.setName("spring2");
repository.save(member2);
Member result = repository.findByName("spring1").get();
Assertions.assertThat(result).isEqualTo(member1);
}
@Test // findAll 테스트
public void findAll() {
Member member1 = new Member();
member1.setName("spring1");
repository.save(member1);
Member member2 = new Member();
member2.setName("spring2");
repository.save(member2);
List<Member> result = repository.findAll();
Assertions.assertThat(result.size()).isEqualTo(2);
}
전체를 Run해보면 에러가 다음과 같은 에러가 뜬다.
그 이유는 테스트 코드는 서로 의존 관계없이(순서에 상관없이) 실행되므로 findAll 메소드에서 이미 repository에 저장이 되어버리고 findByName이 실행되므로 오류가 뜬다.
항상 테스트 코드를 설계할 때는 서로 순서와 관계없이 설계되어야한다.
⭐ 해결 방법으로는 테스트 하나가 끝나면 데이터를 clear 해주면 된다.
MemoryMemberRepository
...
public void clearStore() { // test 코드에서 사용되는데 store에 저장되어있는 데이터를 비워준다.
store.clear();
}
새로 생성해주고
MemoryMemberRepositoryTest
...
@AfterEach // 각 메소드가 실행이 끝날 때마다 동작을 하는 콜백 메소드
public void afterEach() {
repository.clearStore();
}
...
AfterEach 메소드를 넣어주면 각 메소드가 끝날 때마다 store를 clear해준다.
다시 전체를 Run했을 때 오류없이 잘 실행된다.
회원가입 기능과 조회 기능을 만들어보자.
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 -> { // Optional이여서 사용하는 방법(if != null과 같은 의미)
throw new IllegalStateException("이미 존재하는 회원입니다.");
});
}
/**
* 전체 회원을 조회
*/
public List<Member> findMembers() {
return memberRepository.findAll();
}
public Optional<Member> findOne(Long memberId) {
return memberRepository.findByID(memberId);
}
}
클래스 명에서 Alt + Enter를 누르고 Create test를 누르면 해당 클래스의 테스트 환경을 쉽게 만들 수 있다.
class MemberServiceTest {
MemberService memberService;
MemoryMemberRepository memberRepository;
@BeforeEach // 각 메소드를 실행하기 전에 동작하는 메소드
public void BeforeEach() {
memberRepository = new MemoryMemberRepository();
memberService = new MemberService(memberRepository); // 같은 memberRepository를 사용(Dependency Injection)
}
@AfterEach // 각 메소드가 실행이 끝날 때마다 동작을 하는 콜백 메소드
public void afterEach() {
memberRepository.clearStore();
}
@Test
void 회원가입() {
// 테스트 패턴
// given
Member member = new Member();
member.setName("hello");
// when
Long saveId = memberService.join(member);
// then
Member findMember = memberService.findOne(saveId).get();
Assertions.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);
// 예외 처리(member2를 또 join하면 오류가 뜸)
IllegalStateException e = assertThrows(IllegalStateException.class, () -> memberService.join(member2));
Assertions.assertThat(e.getMessage()).isEqualTo("이미 존재하는 회원입니다.");
/*try {
memberService.join(member2);
fail();
} catch (IllegalStateException e) {
// 정상적으로 성공
Assertions.assertThat(e.getMessage()).isEqualTo("이미 존재하는 회원입니다.");
}*/
//then
}
@Test
void findMembers() {
}
@Test
void findOne() {
}
}
결과가 잘 나오는 것을 볼 수 있다.
사용하는 이유
@Controller
, @Service
, @Repository
-> 정형화된 패턴
Controller를 통해서 외부 요청을 받고
Service에서 비지니스 로직을 만들고
Repository에서 데이터를 저장
이렇게 설정을 해주면 Spring이 쭉 가지고 온다.
@Autowired
-> 연결(멤버 컨트롤러가 생성될 때 스프링 빈에 등록되어있는 멤버 서비스 객체를 가져와서 넣어준다. -> 의존관계 주입)
@Service
, @Repository
, @Autowired
이노테이션을 지우고 진행
SpringConfig
@Configuration
public class SpringConfig {
@Bean
public MemberService memberService() {
return new MemberService(memberRepository());
}
@Bean
public MemberRepository memberRepository() {
return new MemoryMemberRepository();
}
}
@Configuration
과 @Bean
을 이용
그림처럼 memberService와 memberRepository를 스프링이 올라올 때 스프링 빈에 올리는 코드이다.
@Controller
는 만들어줘야된다.
HomeController
@Controller
public class HomeController {
@GetMapping("/")
public String home() {
return "home";
}
}
HTML 파일은 깃허브 참고
MemberController
@Controller
public class MemberController {
...
@GetMapping("/members/new") // 회원 등록을 눌렀을 때(Get은 url 창에 치는거와 똑같음, 조회할 때 사용)
public String createForm() {
return "members/createMemberForm"; // 이 위치로 이동(templates에서 찾는다.)
}
@PostMapping("/members/new") // 회원 등록 페이지에서 등록을 눌렀을 때(Post는 데이터를 Form같은 곳에 넣어서 전달할 때 사용, 등록할 때 사용)
public String create(MemberForm form) {
Member member = new Member();
member.setName(form.getName());
memberService.join(member);
return "redirect:/";
}
}
MemberForm
public class MemberForm {
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
HTML 파일은 깃허브 참고
MemberController
@Controller
public class MemberController {
...
@GetMapping("/members")
public String list(Model model) {
List<Member> members = memberService.findMembers(); // 모델에 있는 데이터를 전부 불러옴
model.addAttribute("members", members);
return "members/memberList";
}
}
HTML 파일은 깃허브 참고
위 코드를 실행하고 실행 중인 자바를 내리면 데이터들이 전부 사라지므로 DB를 사용해 해결하자
JPA를 사용하면 SQL과 데이터 중심의 설계에서 객체 중심의 설계로 패러다임을 전환할 수 있고 개발 생산성을 크게 높일 수 있다.
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;
}
}
JpaMemberRepository
public class JpaMemberRepository implements MemberRepository {
private final EntityManager em;
public JpaMemberRepository(EntityManager em) {
this.em = em;
}
@Override
public Member save(Member member) {
em.persist(member);
return member;
}
@Override
public Optional<Member> findByID(Long id) {
Member member = em.find(Member.class, id);
return Optional.ofNullable(member);
}
@Override
public Optional<Member> findByName(String name) {
List<Member> result = em.createQuery("select m from Member m where m.name = :name", Member.class)
.setParameter("name", name)
.getResultList();
return result.stream().findAny();
}
@Override
public List<Member> findAll() {
return em.createQuery("select m from Member m", Member.class)
.getResultList(); // Member Entity를 조회하고 Member 객체 자체를 조회
}
}
SpringConfig
@Configuration
public class SpringConfig {
private EntityManager em;
@Autowired
public SpringConfig(EntityManager em) {
this.em = em;
}
@Bean
public MemberService memberService() {
return new MemberService(memberRepository());
}
@Bean
public MemberRepository memberRepository() {
return new JpaMemberRepository(em);
}
}
Service부분에 @Transactional
해주기
스프링 JPA를 사용하면 인터페이스 이름만으로도 개발이 가능해진다.