프록시
- 실제 클래스를 상속받아서 만들어짐(클래스와 겉모양이 같음)
- em.find()
- 데이터베이스를 통해서 실제 엔티티 객체 조회
- em.getReference()
- 데이터베이스 조회를 미루는 가짜(프록시) 엔티티 객체 조회
- 그러나 해당 객체를 사용할 때는 쿼리가 나감
- Entity target = null
- 프록시 초기화
- 프록시에 값이 없을 때(target = null) 영속성 컨텍스트를 통해 진짜 값을 요청 후 가지고 있음
- 초기화 여부 확인(true, false) : PersistenceUnitUtil().isLoaded(Object entity)
- 조회 등 쿼리가 나감(초기화 이후에는 X, 영속성 컨텍스트를 통해 가져옴)
- 프록시 강제 초기화
- org.hibernate.Hibernate.initialize(entity)
- 강제 호출 (member.getName() 등)
- 프록시 클래스 확인
- entity.getClass().getName()
특징
프록시 객체는 처음 사용할 때 한번만 초기화(초기화 이후에는 영속성 컨텍스트를 통해 가져옴)
초기화 시 프록시 객체가 실제 엔티티로 바뀌는 것은 아님
초기화 이후 프록시 객체를 통해서 실제 엔티티에 접근 가능(프록시 객체 호출 시 실제 객체의 메서드 호출)
프록시 객체는 원본 엔티티를 상속받음, 따라서 타입 체크시 주의(instance of 사용)
Member m1 = em.find(Member.class, member1.getId()); Member m2 = em.getReference(Member.class, member2.getId()); ... private static void temp(Member m1, Member m2){ System.out.println(m1.getClass() == m2.getClass()); //false System.out.println(m1 instanceof Member); //true System.out.println(m2 instanceof Member); //true }
영속성 컨텍스트에 찾는 엔티티가 이미 있으면 em.getReference()를 호출해도 실제 엔티티 반환(반대도 마찬가지)
Member m1 = em.find(Member.class, member1.getId()); Member m2 = em.getReference(Member.class, member2.getId()); ... System.out.println(m1.getClass()); // Member System.out.println(m2.getClass()); // Member ... // 반대의 경우 Member m1 = em.getReference(Member.class, member2.getId()); Member m2 = em.find(Member.class, member1.getId()); ... System.out.println(m1.getClass()); // Proxy System.out.println(m2.getClass()); // Proxy
영속성 컨텍스트의 도움을 받을 수 없는 준영속 상태일 때, 프록시를 초기화하면 문제 발생
Member m = em.getReference(Member.class, member2.getId()); //em.close(); //em.clear(); em.detach(m); System.out.println(m.getUsername());
- em.detach(객체명), em.close(), em.clear() 등 이후 프록시 사용시 예외 발생
'Spring > JPA' 카테고리의 다른 글
영속성 전이 (0) | 2023.12.29 |
---|---|
즉시로딩, 지연로딩 (1) | 2023.12.23 |
MappedSuperclass (0) | 2023.12.20 |
상속관계 매핑 (0) | 2023.12.20 |
연관관계 매핑 (1) | 2023.12.19 |