공부/JPA

애플리케이션 구현 준비

Stair 2025. 6. 30. 09:30
반응형

회원 도메인 개발

구현 기능

- 회원 등록

- 회원 목록 조회

 

순서

- 회원 엔티티 코드 다시 보기

- 회원 리포지토리 개발

- 회원 서비스 개발

- 회원 기능 테스트

 

 

 

회원 리포지토리 개발

@Repository
public class MemberRepository {

    @PersistenceContext
    private 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) 주입

 

기능 설명

- save() : 회원 저장

- findOne() : 저장된 회원 한명을 id값을 통해 조회

- findAll() : 저장된 회원을 List로 모두 조회

- findByName() : 저장된 회원을 이름을 통해 조회

 

 

 

회원 서비스 개발

@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) {
        //Exception
        List<Member> findMembers = memberRepository.findByName(member.getName());
        if (!findMembers.isEmpty()) {
            throw new IllegalStateException("이미 존재하는 회원입니다.");
        }
    }

    //회원 전체 조회
    public List<Member> findMembers() {
        return memberRepository.findAll();
    }

    public Member findOne(Long memberId) {
        return memberRepository.findOne(memberId);
    }
}

 

- @Service

- @Transactional : 트랜잭션, 영속성 컨텍스트

    - readOnly = true : 데이터의 변경이 없는 읽기 전용 메서드에 사용, 영속성 컨텍스트를 플러시 하지 않으므로 약간의 성능 향상(읽기 전용에는 다 적용)

    - 데이터베이스 드라이버가 지원하면 DB에서 성능 향상

- @Autowired

    - 생성자 injection 많이 사용, 생성자가 하나면 생략 가능

 

참고 : 실무에서는 검증 로직이 있어도 멀티 쓰레드 상황을 고려해서 회원 테이블의 회원명 컬럼에 유니크 제약 조건을 추가하는 것이 안전하다.

 

참고 : 스프링 필드 주입 대신 생성자 주입을 사용하자.

 

 

 

회원 기능 테스트

테스트 요구사항

- 회원 가입을 성공해야 한다.

- 회원가입 할 때 같은 이름이 있으면 예외가 발생해야 한다.

 

@SpringBootTest
@Transactional
class MemberServiceTest {

    @Autowired
    MemberService memberService;

    @Autowired
    MemberRepository memberRepository;


    @Test
    @Rollback(value = false)
    public void 회원가입() throws Exception {
        //given
        Member member = new Member();
        member.setName("kim");

        //when
        Long savedId = memberService.join(member);

        //then
        assertEquals(member, memberRepository.findOne(savedId));
    }


    @Test
    public void 중복_회원_예외() throws Exception {
        //given
        Member member1 = new Member();
        member1.setName("kim");

        Member member2 = new Member();
        member2.setName("kim");

        //when
        memberService.join(member1);
        try {
            memberService.join(member2); //예외가 발생해야 한다
        } catch (IllegalStateException e) {
            return;
        }

        //then
        fail("예외가 발생해야 한다.");
    }

}

 

- @SpringBootTest : 스프링 부트를 띄우고 테스트(이 어노테이션이 없으면 @Autowired가 전부 실패된다.)

- @Transactional : 반복 가능한 테스트 지원, 각각의 테스트를 실행할 때마다 트랜잭션을 시작하고 테스트가 끝나면 트랜잭션을 강제로 롤백(이 어노테이션이 테스트 케이스에서 사용될 때만 롤백한다)

 

반응형