이번 글에서는 JPA에서 가장 중요한 개념 중에 하나인 영속성 컨텍스트(Persistence Context)와 이를 다루는 EntityManager를 알아보도록 하겠다.
JPA를 사용하기 위해서는 Database 구조와 맵핑된 JPA Entity 들을 먼저 생성하게 된다.
그리고, 모든 JPA의 동작은 이 Entity들을 기준으로 돌아가게 되는데, 이 때 Entity들을 관리하는 역할을 하는 녀석이 바로 EntityManager인 것이다.
JPA는 기본적으로 한 요청 당, 하나의 EntityManager를 사용한다.
또, 각 EntityManager들은 정해진 영속성 컨텍스트를 참조하게 된다.
그리고 이렇게 만들어진 EntityManager로 데이터를 다루려면 가장 먼저 Entity가 "영속화"되어 있어야 한다.
참고) 일반적으로는 EntityManager 한 개당 하나의 영속성 컨텍스트를 갖지만, 스프링에서는 공통된 영속성 컨텍스트 하나를 여러 EntityManager가 참조한다.
좀 더 쉽게 이야기 한다면, DB에 있던 데이터 또는 사용자가 넘긴 데이터를 Entity로 만든 후, EntityManager의 영속성 컨텍스트에서 들고 있겠다는 것이다.
간단히 정리해보자면, 아래와 같을 것이다.
1. (요청)
2. EntityManager 만들어짐
3. Entity들을 영속성 컨텍스트에 생성 -> Entity 영속화
4. Entity Manager가 영속성 컨텍스트를 기반으로 요청(생성, 조회, 수정, 삭제)처리
이 때, 각 Entity들은 상황에 맞게 생명주기를 갖게 된다.
아래의 이미지는
User
를 생성하는 예제이다.
JPA의 Entity는 영속성 컨텍스트를 기준으로 'Entity가 어떻게 되어있냐?'라는 질문에 대응하는 상태를 갖는다.
그 상태들은 아래와 같다.
위에서 충분히 이야기했지만 다시 정리하자면, 영속성 컨텍스트는 JPA에서 엔티티를 관리하기 위해서 만들어져있는 어떠한 공간이라고 할 수 있겠다.
JPA에서 영속성 컨텍스트라는 DB와 시스템 간의 중개자 역할을 하는 개념을 사용하는 이유는 이러한 구조가 갖는 장점들 때문이다.
어떠한 장점들이 있는지 하나하나 알아보자.
ex) 데이터를 조회하는 기능을 처리할 때, DB에서 바로 조회하는 것이 아니라 영속성 컨텍스트에서 먼저 조회한다.
ex) 영속되어 있는 동일한 id를 갖는 엔티티를 2번 조회한다고 가정해보자. 이 때, 조회된 각각의 객체는 동일한 주소값을 같는다. 즉, 정확히 동일한 객체인 것이다.
ex) 트랜잭션을 commit하는 순간에 영속성 컨텍스트의 엔티티에 대한 쿼리를 DB에 실행한다.
ex) 영속된 엔티티를 수정하면, 트랜잭션이 commit되는 순간 변경 쿼리가 DB에 실행된다.
EntityManager의 .flush()
메서드가 실행되게 되면 영속성 컨텍스트의 쓰기 지연 저장소에 존재하는 쿼리들을 DB에 수행하게 된다.
즉, flush()
는 영속성 컨텍스트의 변경사항을 DB와 동기화하는 것이라고 볼 수 있다.
그렇다고, .flush()
를 실행시켰다고 수행된 쿼리가 commit되는 것은 아니고, commit을 직접해줘야한다.
단, EntityManager로 트랜잭션을 생성한 경우, commit을 하면 .flush()
메서드가 자동적으로 먼저 실행된다.