JPA

기본키 매핑 전략에 따른 INSERT QUERY 실행 시점

DeJa 2021. 12. 28. 21:06
728x90

기본키 매핑 전략에 따른 INSERT QUERY 실행 시점

JPA 의 기본키 생성 전략 중, IDENTITY 와 SEQUENCE 전략을 사용할 때, 실제 언제 INSERT QUERY 가 나가고 어떻게 영속성 엔티티에 들어가서 관리되는지 확인해보자.

IDENTITY

@DisplayName("IDENTITY 전략 테스트")
@SpringBootTest
class IdentityStrategyTest {

    @Value("${persistence.unitname}")
    private String persistenceUnitName;

    @DisplayName("em.persist() 시 INSERT 쿼리가 나가는지 테스트")
    @Test
    void insertQueryIsWorkAtPersist() throws Exception {
        EntityManagerFactory emf = Persistence.createEntityManagerFactory(persistenceUnitName);
        EntityManager em = emf.createEntityManager();
        EntityTransaction tx = em.getTransaction();

        tx.begin();
        try {
            Member member = new Member();
            member.setUsername("BAEK");

            // 이 시점에, INSERT QUERY 가 실행되고
            // 내부적으로 DB 에 등록된 ID 값을 가져와서 MEMBER 엔티티에 설정한다.
            // 따라서, MEMBER 엔티티에 ID(PK) 값이 있기 때문에 1차 캐시(영속성 엔티티)에서 관리될 수 있다.
            em.persist(member);

            // MEMBER 엔티티에서 ID 값을 꺼내는 순간 ID(PK) 값이 바인딩 되는것이 아니다.
            System.out.println(member.getId());

            // 1차 캐시에서 조회
            Member findMember = em.find(Member.class, 1L);
            System.out.println(findMember.getId());
            tx.commit();
        } catch (Exception e) {
            tx.rollback();
        } finally {
            em.close();
            emf.close();
        }
    }
}

em.persist(member) 시점에 아래와 같은 INSERT QUERY 가 나가는 것을 볼 수 있다.

Hibernate: 
    /* insert com.jtcwp.purejpa.domain.member.Member
        */ insert 
        into
            Member
            (name) 
        values
            (?)

SEQUENCE

@DisplayName("SEQUENCE 전략 테스트")
@SpringBootTest
class SequenceStrategyTest {

    @Value("${persistence.unitname}")
    private String persistenceUnitName;

    @DisplayName("시퀀스를 통한 기본키 생성 전략을 사용하는 경우 INSERT 쿼리가 트랜잭션 커밋 시점에 나가는지 테스트")
    @Test
    void insertQueryIsWorkAtTransactionCommit() throws Exception {
        EntityManagerFactory emf = Persistence.createEntityManagerFactory(persistenceUnitName);
        EntityManager em = emf.createEntityManager();
        EntityTransaction tx = em.getTransaction();

        tx.begin();
        try {
            Member member = new Member();
            member.setUsername("BAEK");

            // 이 시점에, 시퀀스가 호출되어서(call next value for hibernate_sequence) 
            // 데이터베이스에 생성된 시퀀스 값을 MEMBER 엔티티 ID 에 넣어서, 영속성 엔티티로 등록 한다.
            em.persist(member);

            // MEMBER 엔티티에서 ID 값을 꺼내는 순간 ID(PK) 값이 바인딩 되는것이 아니다.
            System.out.println(member.getId());

            // 1차 캐시에서 조회
            Member findMember = em.find(Member.class, 1L);
            System.out.println(findMember.getId());

            // 이 시점에 실제 INSERT QUERY 가 나간다.
            tx.commit();
        } catch (Exception e) {
            tx.rollback();
        } finally {
            em.close();
            emf.close();
        }
    }
}

em.persist(member) 시점에 아래와 같이 시퀀스가 호출되는 것을 볼 수 있다.

Hibernate: 
    call next value for hibernate_sequence

테스트에서 확인한 것 처럼, 기본키 매핑에 어떠한 전략을 사용하는지에 따라 INSERT QUERY 가 실행되는 시점이 달라지기 때문에 잘 알고 있어야 한다.

728x90