1. 프록시
◍ 기초
○ em.find = 데이터 베이스 통해서 실제 엔티티 객체 조회
○ em.getReference(): 데이터 베이스 조회를 미루는 가짜 엔티티 조회(프록시)
○ 특징
◼ 실제 클래스를 상속 받아서 만들어짐.
◼ 실제 클래스와 겉 모양이 같다.
◼ 사용하는 입장에서는 진짜 객체인지 프록시 객체인지 구분하지 않고 사용하면 됨(이론상)
◼ 프록시 객체는 실제 객체의 참조(target)를 보관
◼ 프록시 객체를 호출하면 프록시 객체는 실제 객체의 메소드 호출
○ 프록시 객체의 초기화
Member member = em.getReference(Member.class, "id1");
member.getName()
//1. getName()
//2. 영속성 컨텍스트에 초기화 요청
//3. DB조회
//4. 실제 Entity 생성
//5. target.getName();
◼ 처음 사용할때 한번만 초기화 (두번 세번 X)
◼ 초기화시 프록시 객체가 실제 엔티티로 바뀌는 것 아님. 초기화시 프록시 객체 통해 실제 엔티티 접근 가능
◼ 프록시 객체는 원본 엔티티를 상속받음. 그래서 타입 체크시 주의해야함. (== 안됨. instance of 사용) <member 1,2>
◼ 영속성 컨텍스트에 찾는 엔티티가 이미 있으면 em.getReference()호출해도 실제 엔티티 반환. <이미 find, member 1>
(== 맞추어 주어야 함.)
◼ 영속성 컨텍스트의 도움을 받을 수 없는 준영속 상태일 때, 프록시를 초기화하면 문제 발생.
(하이버네이트 org.hibernate.LazyInitializationException 터트림)
○ 확인
◼ 프록시 인스턴스의 초기화 여부
◻ PersiistenceUnitUtil.isLoaded(Object entity)
◼ 프록시 클래스 확인 방법
◻ org.getClass().getName() 출력 (..javasist.. or HibernateProxy..)
◼ 프록시 강제 초기화
◻ org.hibernate.Hibernate.initialize(entity);
◼ 강제 호출
◻ member.getName()
2. 즉시로딩과 지연로딩
◍ 즉시로딩 (EAGER_
○ 즉시로딩 주의
◼ 가급적 지연 로딩만 사용(특히 실무에서)
◼ 즉시 로딩 적용시 예상치 목한 SQL 발생
◻ find할때 모두 조인을 해야함. 쿼리가 엄청 커짐. (성능 이슈)
◼ 즉시 로딩은 JPQL에서 N+1 문제 일으킴
◻ 쿼리 하나당 추가 쿼리가 나가야 되는 것.
◼ @ManyToOne, @OneToOne은 기본이 즉시 로딩 -> LAZY 로 설정
◼ @OneToMany, @ManyToMany는 기본이 지연로딩.
◍ 지연로딩
○ 프록시를 만들었다가 실제 team 사용하는 시점에 초기화(DB조회).
○ 1. member1 로딩 2.지연로딩
○ 실무
◼ 모든 연관관계에 지연 로딩을 사용해라!
◼ 실무에서 즉시로딩 사용 금지.
◼ JPQL fetch 조인이나, 엔티티 그래프 기능 사용
◼ 즉시 로딩 상상하지 못한 쿼리 나감.
3; 영속성 전이:CASCADE
◍ 특정 엔티티를 영속 상태고 만들 때 연관된 엔티티도 함께 연속 상태로 만들고 싶을때. (부모 저장때 자식도 저장.)
◍ 영속성 전이는 연관관계를 매핑하는 것과 상관 없음.
◍ 엔티티를 영속활 할때 연관된 엔티티도 함께 영속화하는 편리함을 제공.
◍ 엔티티의 소유자가 하나일떄.
◍ 종류
○ ALL: 모두 적용
○ PERSIST: 영속
○ REMOVE: 삭제
○ MERGE: 병합
○ REFRESH: refresh
○ MERGE: merge
4.고아객체
◍ 고아객체 제거: 부모 엔터티와 연관관계가 끊어진 자식 엔티티 자동으로 삭제
◍ orphanRemoval = true
◍ 주의
○ 참조가 제거된 엔티티는 다른 곳에서 참조하지 않는 고아 객체로 보고 삭제하는 기능
○ 참조하는 곳이 하나일때 사용해야함.
○ 특정 엔티티가 개인 소유할 때 사용
○ @OneToOne, @OneToMany만 가능
○ 개념적으로 부모 제거시 자식 보아. 그래서 고아 객체 제거 기능 활성화시 부모 제거시 자식도 제거.
5.통합
◍ 3,4 옵션을 모두 활성화 하면 부모 엔티티를 통해ㅔㅐ서 자식의 생명 주기를 관리 할 수 있음.
◍ 도메인 주도 설계의 Aggregate Root 개념 구현 시 유용
◍ 스스로 생명주기를 관리하는 엔티티는 em.persist()로 영속화, em.remove()로 제거
'Spring > JPA' 카테고리의 다른 글
JPA Basic - JPQL Basic (0) | 2024.01.10 |
---|---|
JPA Basic - 값 타입 (0) | 2024.01.09 |
JPA Basic - 상속관계 매핑 (0) | 2024.01.09 |
JPA Basic - 엔티티 매핑 (0) | 2024.01.09 |
JPA Basic - 영속성 관리 (0) | 2024.01.08 |