[작성일: 2023. 08. 24]

회원 리포지토리 개발
@Repository @RequiredArgsConstructor public class MemberRepository { private final EntityManager em; public void save(Member member) { // 회원 가입 em.persist(member); } public Member findOne(Long id) { // 회원 조회 return em.find(Member.class, id); } public List<Member> findAll() { // 모든 회원 조회 return em.createQuery("select m from Member m", Member.class) .getResultList(); } public List<Member> findByName(String name) { // 회원 이름으로 조회 return em.createQuery("select m from Member m where m.name = :name", Member.class) .setParameter("name", name) .getResultList(); } }
- @Repository: 스프링 빈으로 등록하여 JPA 예외를 스프링 기반 예외로 예외 변환한다.
- @PersistenceContext: 엔티티 매니저를 주입한다.(EntityManager)
- @PersistenceUnit: 엔티티 매니저 팩토리를 주입한다.(EntityManagerFactory)
참고: 스프링 데이터 JPA를 사용하면 EntityManager도 주입이 가능하다.
회원 서비스 개발
@Service @Transactional(readOnly = true) @RequiredArgsConstructor public class MemberService { private final MemberRepository memberRepository; @Transactional(readOnly = false) // 회원 가입 public Long join(Member member) { validateDuplicateMember(member); // 중복 회원 검증 memberRepository.save(member); return member.getId(); } private void validateDuplicateMember(Member member) { List<Member> findMembers = memberRepository.findByName(member.getName()); if (!findMembers.isEmpty()) { throw new IllegalStateException("이미 존재하는 회원입니다."); } } // 회원 전체 조회 public List<Member> findMembers() { return memberRepository.findAll(); } // 회원 1명 조회 public Member findOne(Long id) { return memberRepository.findOne(id); } }
- @Service
- @Transactional: 트랜잭션, 영속성 컨텍스트
- readOnly = true: 데이터의 변경이 없는 읽기 전용 메서드에 사용, 영속성 컨텍스트를 플러시 하지 않으므로 약간 성능이 향상한다.(읽기 전용에는 전부 적용한다.)
- 데이터베이스 드라이버가 지원하면 DB에서 성능 향상
- @Autowired: 생성자 주입으로 많이 사용하며 생성자가 하나면 생략이 가능하다.
참고: 실무에서는 검증 로직이 있어도 멀티 스레드 상황을 고려해서 회원 테이블의 회원명 컬럼에 유니크 제약 조건을 추가하는 것이 안전하다. 그리고 스프링 필드 주입 대신에 생성자 주입을 사용하자.
- 필드주입 말고 생성자 주입 방식을 권장한다.
- 변경 불가능한 안전한 객체 생성이 가능하다.
- 생성자가 하나면 @Autowired를 생략할 수 있다.
- final 키워드를 추가하면 컴파일 시점에 memberRepository를 설정하지 않는 오류를 체크할 수 있다.
회원 기능 테스트
테스트 요구사항
- 회원가입을 성공해야 한다.
- 회원가입 할 때 같은 이름이 있으면 예외가 발생해야 한다.
회원가입 테스트
@SpringBootTest @RunWith(SpringRunner.class) @Transactional public class MemberServiceTest { @Autowired MemberService memberService; @Autowired MemberRepository memberRepository; @Test @Rollback(value = false) public void 회원가입() throws Exception { // given Member member = new Member(); member.setName("j"); // when Long saveId = memberService.join(member); // then assertEquals(member, memberRepository.findOne(saveId)); } @Test(expected = IllegalStateException.class) public void 중복_회원_예외() throws Exception { // given Member member1 = new Member(); member1.setName("j"); Member member2 = new Member(); member2.setName("j"); // when memberService.join(member1); memberService.join(member2); // 예외가 발생해야 함. // then fail("예외가 발생해야 한다."); } }
- @RunWith(Springrunner.class): 스프링과 테스트 통합
- @SpringBootTest: 스프링 부트를 띄우고 테스트(이게 없으면 @Autowired 전부 실패)
- @Transactional: 반복 가능한 테스트 지원, 각각의 테스트를 실행할 때마다 트랜잭션을 시작하고 테스트가 끝나면 트랜잭션을 강제로 롤백(이 어노테이션이 테스트 케이스에서 사용될 때만 롤백)
🐣 출처: 인프런 김영한님 강의
이 글은 인프런의 김영한님 JPA 강의를 보고 작성한 글입니다.
강의를 들으면서 정리한 글이므로 틀린 내용이나 오타가 있을 수 있습니다.