[자바 표준 ORM JPA]3장 영속성 관리

엔티티 매니저 팩토리와 엔티티 매니저

엔티티 매니저 펙토리는 여러 스레드가 동시에 접근해도 안전하므로 서로 다른 스레드 간에 공유해도 되지만, 엔티티 매니저는 여러 스레드가 동시에 접근하면 동시성 문제가 발생 하므로 스레드 간에 절대 공유하면 안 된다. image

위 그림을 보면 하나의 EntityManagerFactory에서 다수의 EntityManager를 생성했다. EntityManager1은 아직 데이터베이스 커넥션을 사용하지 않는데, 엔티티 매니저는 데이터베이스 연결이 꼭 필요한 시점까지 커넥션을 얻지 않는다. EntityManager2는 커넥션을 사용 중인데 보통 트랜잭션을 시작할 때 커넥션을 획득한다.

영속성 컨택스트란?

영속성 컨택스트는 “엔티티를 영구 저장하는 환경” em.persist(member); 에서 persist() 메소드는 엔티티 매니저를 사용해서 회원 엔티티를 영속성 컨텍스트에 저장한다.

엔티티 생명주기

image

엔티티에는 4자지 상태가 존재한다.

  • 비영속 new/transient: 영속성 컨텍스트와 전혀 관계가 없는 상태
  • 영속 managed: 영속성 컨텍스트에 저장된 상태
  • 준영속 detached: 영속성 컨텍스트에 저장되었다가 분리된 상태
  • 삭제 removed: 삭제된 상태

    비영속

    엔티티 객체를 생성했다. 순수한 객체 상태이며 아직 저장하지 않았다.

    영속

    영속성 컨텍스트가 관리하는 엔티티를 영속 상태라 한다. 결국 영속 상태라는 것은 영속성 컨텍스트에 의 관리된다는 뜻이다.

    준영속

    영속성 컨텍스트가 관리하던 영속 상태의 엔티티를 영속성 컨텍스트가 관리하지 않으면 준영속 상태가 된다.

    삭제

    엔티티를 영속성 컨텍스트와 데이터베이스에서 삭제한다.

영속성 컨텍스트의 특징

  • 영속성 컨텍스트와 식별자 값 : 영속 상태는 식별자 값(기본키와 맵핑한 값)이 반드시 있어야 한다.
  • 영속성 컨텍스트와 데이터베이스 저장
  • 영속성 컨텍스트가 엔티티를 관리하면 다음과 같은 점이 있다.
    • 1차 캐시
    • 동일성 보장
    • 트랙잭션을 지원하는 쓰기 지연
    • 변경 감지
    • 지연 로딩

조회

image
image

1차 캐시에서 식별자 값으로 엔티티를 찾는다. 만약 찾는 엔티티가 있으면 데이터베이스를 조회하지 않고 메모리에 있는 1차 캐시에서 엔티티를 조회한다. 만약 엔티티가 1차 캐시에 없으면 엔티티 매니저는 데이터베이스를 조회해서 엔티티를 생성한다. 그리고 1차 캐시에 저장한 후에 영속 상태의 엔티티를 반환한다. (영속성 컨텍스트는 성능상 이점과 엔티티의 동일성을 보장한다.)

등록

image
먼저 회원 A를 영속화 했다. 영속성 컨텍스트는 1차 캐시에 회원 엔티티를 저장하면서 동시에 회원 엔티티 정보로 등록 쿼리를 만든다. 그리고 만들어진 등록 쿼리를 쓰기 지연 SQL 저장소에 보관한다.

image
다음으로 회원 B를 영속화했다. 마찬가지로 회원 엔티티 정보로 등록 쿼리를 생성해서 쓰기 지연 SQL 저장소에 보관한다. 현재 쓰기 지연 SQL 저장소에는 등록 쿼리가 2건 저장되었다.

image

마지막으로 트랜잭션을 커밋했다. 트랜잭션을 커밋하면 엔티티 매니저는 우선 영속성 컨텍스트를 플러시한다. 플러시는 영속성 컨텍스트의 변경 내용을 데이터베이스에 동기화하는 작업인데 이때 등록, 수정, 삭제한 엔티티를 데이터베이스에 반영한다. 좀 더 자세히 말하면 쓰기 지연 SQL 저장소에 모인 쿼리를 데이터베이스에 보낸다.

JPA는 엔티티를 영속성 컨텍스트에 보관할 때, 최초 상태를 복사해서 저장해두는데 이것을 스냅샷이라고 한다. 그리고 플러시 시점에 스냅샷과 엔티티를 비교해서 변경된 엔티티를 찾는다.

  1. 트랜잭션을 커밋하면 엔티티 매니저 내부에서 먼저 플러시flush()가 호출된다.
  2. 엔티티와 스냅샷을 비교해서 변경된 엔티티를 찾는다.
  3. 변경된 엔티티가 있으면 수정 쿼리를 생성해서 쓰기 지연 SQL 저장소에 보낸다.
  4. 쓰기 지연 저장소의 SQL을 데이터베이스에 보낸다.
  5. 데이터베이스 트랜잭션을 커밋한다.

삭제

삭제 대상을 조회한 후 remove로 삭제를 한다. 삭제된 대상은 갈비지 컬렉션의 대상이 되도록 두는 것이 좋다

플러쉬(flush

플러시flush()는 영속성 컨텍스트의 변경 내용을 데이터베이스에 반영한다)

  1. 변경 감지가 동작해서 영속성 컨텍스트에 있는 모든 엔티티를 스냅샷과 비교해서 수정된 엔티티를 찾는다. 수정된 엔티티는 수정 쿼리를 만들어 쓰기 지연 SQL 저장소에 등록한다.
  2. 쓰기 지연 SQL 저장소의 쿼리를 데이터베이스에 전송한다(등록, 수정, 삭제, 쿼리) 영속성 컨태스트를 플러시 하는방법 3가지
    • 직접호출
      • 메소드를 직접 호출해서 영속성 컨텍스트를 강제로 플러시한다.
      • 테스트나 다른 프레임워크와 JPA를 함께 사용할 때를 제외하거 거의 사용하지 않는다.
    • 틀랜잭션 커밋 시 플러시 자동 호출
      • 트랜잭션을 커밋하기 전에 꼭 플러시를 호출해서 영속성 컨택스트의 변경 내용을 데이터베이스에 반영해야한다.
    • JPQL 쿼리 실행시 플러시 자동 호출
      • 쿼리를 실행하기 직전에 플러시를 해서 변경 내용을 데이터베이스에 반영해야 한다.
      • JPA는 이런 문제를 예방하기 위해 JPQL을 실행할 때도 플러시를 자동 호출한다.
      • 참고로 식별자를 기준으로 조회하는 find() 메서드를 호출할 때는 플러시가 실행되지 않는다.

준영속

영속성 컨텍스트가 관리하는 영속 상태의 엔티티가 영속성 컨텍스트에서 분리detached된 것을 준영속 상태라 한다. 따라서 준영속 상태의 엔티티는 영속성 컨텍스트가 제공하는 기능을 사용할 수 없다.

영속 상태의 엔티티를 준영속 상태로 만드는 방법은 크게 3가지이다.

em.detach(entity)

: 특정 엔티티만 준영속 상태로 전환한다. image
image
준영속 상태는 영속성 컨텍스트로부터 분리된 상태다

em.clear()

: 영속성 컨텍스트를 완전히 초기화한다.

  • clear()는 영속성 컨텍스트에 있는 모든 것을 영속성 컨텍스트를 제거하고 새로 만든 것처럼 초기화 시킨다.
  • 준영속 상태이므로 영속성 컨텍스트가 지원하는 변경 감지는 동작하지 않는다
  • 따라서 데이터베이스에도 반영되지 않는다.

    em.close()

    : 영속성 컨텍스트를 종료한다.

  • 영속성 컨텍스트를 종료하면 해당 영속성 컨텍스트가 관리하던 영속 상태의 엔티티가 모두 준영속 상태가 된다.

Categories:

Updated:

Leave a comment