본문 바로가기

DB

JPA - 프록시

프록시

프록시 기초

  • em.find() vs em.getReference()
  • em.find() : 데이터베이스를 통해서 실제 엔티티 객체 조회
  • em.getReference(): 데이터베이스 조회를 미루는 가짜(프록시) 엔티티 객체 조회

프록시 특징

  • 실제 클래스를 상속 받아서 만들어짐
  • 실제 클래스와 겉 모양이 같다.
  • 사용하는 입장에서는 진짜 객체인지 프록시 객체인지 구분하지 않고 사용하면 됨(이론상)

  • 프록시 객체는 실제 객체의 참조(target)를 보관
  • 프록시 객체를 호출하면 프록시 객체는 실제 객체의 메소드 호출

프록시 객체의 초기화

Member member = em.getReference(Member.class, "id1");
member.getName();

 

코드 실행 결과

Team team = new Team();
team.setName("TeamA");
em.persist(team);

Member member = new Member();
member.setName("harris");
member.setTeam(team);
em.persist(member);

em.flush();
em.clear();

위와 같이 Team과 Member를 저장하고 영속성 컨텍스트를 비워줬다.

이 때 em.find(Member.class, member.getId()) 의 결과와 em.getReference() 의 결과가 어떻게 다른지 먼저 확인해보자.

Member findMember = em.find(Member.class, member.getId());
System.out.println("findMember = " + findMember.getClass());

// 결과
Hibernate: 
    select
        member0_.MEMBER_ID as MEMBER_I1_0_0_,
        member0_.name as name2_0_0_,
        member0_.TEAM_ID as TEAM_ID3_0_0_ 
    from
        Member member0_ 
    where
        member0_.MEMBER_ID=?
findMember = class hellojpa.Member
  • hibernate에서 바로 select 쿼리가 나감을 확인 할 수 있다.
  • findMember 의 클래스를 확인했을 때 Member class 임을 확인할 수 있다.
Member refMember = em.getReference(Member.class, member.getId());
System.out.println("refMember = " + refMember.getClass());

// 결과
refMember = class hellojpa.Member$HibernateProxy$dbqU4oAg
  • em.getReference() 를 호출했을 때에는 select 쿼리가 발생하지 않았다.
  • 또한 refMember 의 클래스 타입은 Proxy 객체임을 확인할 수 있다.
  • refMember.getName() 함수를 호출하면?
    • 그때 select 쿼리가 발생함을 확인할 수 있다.

프록시의 특징

  • 프록시 객체는 처음 사용할 때 한 번만 초기화
  • 프록시 객체를 초기화 할 때, 프록시 객체가 실제 엔티티로 바뀌는 것은 아님, 초기화되면 프록시 객체를 통해서 실제 엔티티에 접근 가능
  • 프록시 객체는 원본 엔티티를 상속받음, 따라서 타입 체크시 주의해야함 (== 비교 실패, 대신 instance of 사용)
  • 영속성 컨텍스트에 찾는 엔티티가 이미 있으면 em.getReference()를 호출해도 실제 엔티티 반환 (반대도 마찬가지!!)
Member refMember = em.getReference(Member.class, member.getId());
Member findMember = em.find(Member.class, member.getId());
System.out.println("refMember = " + refMember.getClass());
System.out.println("findMember = " + findMember.getClass());
System.out.println("findMember == refMember : " + (refMember == findMember));
//결과
refMember = class hellojpa.Member$HibernateProxy$Iu06xoKG
findMember = class hellojpa.Member$HibernateProxy$Iu06xoKG
findMember == refMember : true

Member findMember = em.find(Member.class, member.getId());
Member refMember = em.getReference(Member.class, member.getId());
System.out.println("refMember = " + refMember.getClass());
System.out.println("findMember = " + findMember.getClass());
System.out.println("findMember == refMember : " + (refMember == findMember));
//결과
refMember = class hellojpa.Member
findMember = class hellojpa.Member
findMember == refMember : true
  • 영속성 컨텍스트의 도움을 받을 수 없는 준영속 상태일 때, 프록시를 초기화하면 문제 발생
    (하이버네이트는 org.hibernate.LazyInitializationException 예외를 터트림)
    • 프록시가 아니라 실제 엔티티라면??
Member refMember = em.getReference(Member.class, member.getId());
em.flush();
em.clear();
System.out.println("refAgain : " + refMember.getName());
//결과
org.hibernate.LazyInitializationException: could not initialize proxy [hellojpa.Member#2] - no Session
  • 아직 객체를 가져오지 않은 proxy인 상태에서 영속성 컨텍스트가 비워지면 객체를 조회시 LazyInitializationException 이 발생한다. 말 그대로 지연 초기화가 실패했을 때 발생한다.
    • 프록시가 아닌 실제 엔티티라면, 다시 말해 LAZY 가 아닌 EAGER 로 받아온다면 어떨까?
Member findMember = em.find(Member.class, member.getId()); //이 시점에 조회해서 객체 초기화
em.flush();
em.clear();
System.out.println("findAgain : " + findMember.getName());
//결과
Hibernate: 
    select
        member0_.MEMBER_ID as MEMBER_I1_0_0_,
        member0_.name as name2_0_0_,
        member0_.TEAM_ID as TEAM_ID3_0_0_ 
    from
        Member member0_ 
    where
        member0_.MEMBER_ID=?
findAgain : harris
  • 결과는 객체를 조회한 시점에서 초기화하기 때문에 Exception이 발생하지 않는다.

프록시 확인

  • 프록시 인스턴스의 초기화 여부 확인
    emf.getPersistenceUtil.isLoaded(Object entity)
  • 프록시 클래스 확인 방법
    entity.getClass().getName()츨력(..javasist.. or HibernateProxy...)
  • 프록시 강제 초기화
    org.hibernate.Hibernate.initialize(entity);
  • 참고: JPA 표준은 강제 초기화 없음
    강제 호출: member.getName()

'DB' 카테고리의 다른 글

Real MySQL 4장 아키텍처  (0) 2022.05.24
Real MySQL 1장 ~ 3장  (0) 2022.05.23
JPA - @MappedSuperclass  (0) 2021.09.11
JPA - 다양한 연관관계 매핑  (0) 2021.09.11
JPA - 양방향 연관관계와 연관관계의 주인  (0) 2021.09.11