이 글은 김영한님의 자바 ORM 표준 JPA 프로그래밍 - 기본편 강의를 듣고 정리한 글입니다.
JPA에서 가장 중요한 두 가지
//객체를 생성한 상태(비영속)
Member member = new Member();
member.setId("member1");
member.setUsername("회원1");
EntityManager em = emf.createEntityManager();
em.getTransaction().begin();
//객체를 저장한 상태(영속)
em.persist(member);
em.detach(member); // 특정 엔티티만 준영속 상태로 전환
em.clear(); // 영속성 컨텍스트를 완전히 초기화
em.close(); // 영속성 컨텍스트를 종료
em.remove(member);
영속화되었다고 DB에 날라가는 것이 아님!
// code // 비영속 Member member = new Member(); member.setId(100L); member.setName("HelloJPA"); // 영속 System.out.println("===== BEFORE ====="); em.persist(member); System.out.println("===== AFTER ====="); tx.commit();
⇒ 트랜잭션을commit()
하는 시점에서 쿼리가 날라감
Member member = new Member();
member.setId("member1");
member.setUsername("회원1");
// 1차 캐시에 저장됨
em.persist(member);
// 1차 캐시에서 조회 -> 반환
Member findMember = em.find(Member.class, "member1");
// 1차 캐시에서 없는 경우,
// DB 조회 -> **1차 캐시에 저장** -> 반환
Member findMember = em.find(Member.class, "member2");
Member a = em.find(Member.class, "member1");
Member b = em.find(Member.class, "member1");
System.out.println(a == b); // 동일성 비교: true
commit()
하는 순간 데이터베이스에 보내진다. (<property name="hibernate.jdbc.batch_size" value="10"/>
을 통해 batch size를 지정할 수도 있다) EntityManager em = emf.createEntityManager();
EntityTransaction transaction = em.getTransaction();
//엔티티 매니저는 데이터 변경시 트랜잭션을 시작해야 한다.
transaction.begin(); // [트랜잭션] 시작
em.persist(memberA);
em.persist(memberB);
//여기까지 INSERT SQL을 데이터베이스에 보내지 않는다.
//커밋하는 순간 데이터베이스에 INSERT SQL을 보낸다.
transaction.commit(); // [트랜잭션] 커밋
Member member1 = em.find(Member.class, 150L);
member1.setName("AAAAA");
em.persist(member1);
System.out.println("========================");
Member member1 = em.find(Member.class, 151L);
member1.setName("BBBBB");
// persist()를 사용할 필요가 없다
→ JPA는 자바 컬렉션처럼 동작한다. 값을 변경하면 트랜잭션이 commit되는 시점에 변경을 반영한다.
플러시(flush)
- 영속성 컨텍스트의 변경 내용을 데이터베이스에 동기화 (영속성 컨텍스트를 비우지 않음)
- 트랜잭션이라는 작업 단위가 중요하다.
→ 커밋 직전에만 동기화하면 됨. (영속성 컨텍스트 또한 트랜잭션과 주기를 맞춰 설계가 되어야 한다.)플러시 발생
- 변경 감지
- 수정된 엔티티 쓰기 지연 SQL 저장소에 등록 (1차 캐시가 지워지는 것이 아님)
- 쓰기 지연 SQL 저장소의 쿼리를 데이터베이스에 전송 (등록, 수정, 삭제 쿼리)영속성 컨텍스트를 플러시하는 방법
-em.flush()
: 직접 호출
- 트랜잭션 커밋 : 자동 호출
- JPQL 쿼리 실행 : 자동 호출 - DB에서 가져올 것이 없는 경우를 방지하기 위해Member member = new Member(200L, "member200"); em.persist(member); em.flush(); System.out.println("========================"); tx.commit();
commit()
명령어 이전에 INSERT 쿼리가 실행됨