728x90
들어가기 전에
참고 자료 - JPA 구동 방식
영속성 컨텍스트
"엔티티를 영구 저장하는 환경" 이라는 뜻이다. 논리적인 개념이므로 눈에 보이지는 않는다.
일종의 공간이라고 생각하면 된다.
엔티티를 엔티티 매니저를 통해서 영속성 컨텍스트에 접근한다.
EntityManager.persist(entity);
엔티티의 생명주기
- 비영속(new/transient) : 영속성 컨텍스트와 전혀 관계가 없는 새로운 상태
//객체를 생성한 상태(비영속)
Member member = new Member();
member.setId("member1");
member.setUsername("회원1");
- 영속(managed) : 영속성 컨텍스트에 관리되는 상태
//객체를 생성한 상태(비영속)
Member member = new Member();
member.setId("member1");
member.setUsername(“회원1”);
EntityManager em = emf.createEntityManager();
em.getTransaction().begin();
//객체를 저장한 상태(영속)
em.persist(member);
- 준영속(detached) : 영속성 컨텍스트에 저장되었다가 분리된 상태
//회원 엔티티를 영속성 컨텍스트에서 분리, 준영속 상태
em.detach(member);
- 삭제(removed) : 삭제된 상태
//객체를 삭제한 상태(삭제)
em.remove(member);
영속성 컨텍스트의 이점
1. 1차 캐시
영속 컨텍스트는 내부에 1차 캐시를 두고 있다.
//엔티티를 생성한 상태(비영속)
Member member = new Member();
member.setId("member1");
member.setUsername("회원1");
//엔티티를 영속
em.persist(member);
1차 캐시에서 조회
Member member = new Member();
member.setId("member1");
member.setUsername("회원1");
//1차 캐시에 저장됨
em.persist(member);
//1차 캐시에서 조회
Member findMember = em.find(Member.class, "member1");
데이터베이스에서 조회
Member findMember2 = em.find(Member.class, "member2");
2. 동일성 보장
1차 캐시로 반복 가능한 읽기(Repeatable Read) 등급의 트랜잭션 격리 수준을 데이터베이스가 아닌 애플리케이션 차원에서 제공한다.
Member a = em.find(Member.class, "member1");
Member b = em.find(Member.class, "member1");
System.out.println(a == b); //동일성 비교 true
3. 트랜잭션을 지원하는 쓰기 지연
엔티티 등록
EntityManager em = emf.createEntityManager();
EntityTransaction transaction = em.getTransaction();
//엔티티 매니저는 데이터 변경시 트랜잭션을 시작해야 한다.
transaction.begin(); // [트랜잭션] 시작
em.persist(memberA);
em.persist(memberB);
//여기까지 INSERT SQL을 데이터베이스에 보내지 않는다.
//커밋하는 순간 데이터베이스에 INSERT SQL을 보낸다.
transaction.commit(); // [트랜잭션] 커밋
4. 변경 감지 (Dirty Checking)
엔티티 수정
EntityManager em = emf.createEntityManager();
EntityTransaction transaction = em.getTransaction();
transaction.begin(); // [트랜잭션] 시작
// 영속 엔티티 조회
Member memberA = em.find(Member.class, "memberA");
// 영속 엔티티 데이터 수정
memberA.setUsername("hi");
memberA.setAge(10);
//em.update(member) 이런 코드가 있어야 하지 않을까?
transaction.commit(); // [트랜잭션] 커밋
//em.update(member) 이런 코드가 있어야 하지 않을까? 에 대한 답변은 NO!
스냅샷과 엔티티를 비교하여 변경된 부분이 있으면 감지하여 자동 반영한다.
그러므로 따로 update를 해줄 필요가 없다.
728x90
'💻dev > 🌱Java+Spring' 카테고리의 다른 글
JPA | 엔티티 매핑 : @Entity, @Table, 데이터베이스 스키마 자동 생성 (0) | 2023.04.27 |
---|---|
JPA | 플러시와 준영속 상태를 알아보자 (0) | 2023.04.27 |
JPA | JPA란 무엇인가? (0) | 2023.04.27 |
Java | 인터페이스(Interface)를 사용하는 이유와 예제 (0) | 2023.04.10 |
Spring | 인프런 김영한님의 스프링 입문 강의 정리 - Test 편 (0) | 2023.04.07 |