본문 바로가기
이론/JPA

[JPA] Proxy 프록시

by 6161990 2021. 9. 18.

01. Proxy 프록시가 필요한 상황

📍 Member 객체를 조회할 때 속해져있는 Team 도 같이 조회해야할까?

public void printUserAndTeam(String memberId) {
         Member member = entityManager.find(Member.class, memberId);
         Team team = member.getTeam();
         System.out.println("회원 이름" + member.getUserName());
         System.out.println("소속팀" + team.getName());
}
  • Team은 어쩌다 한 번 필요해서 회원을 조회할 때 마다 매번 같이 조회하는 것은 낭비인 상황이다. 
  • 이런 경우 보통 Member 클래스의 Team 필드에 아래와 같은 어노테이션을 지정한다. 
@ManyToOne(fetch = FetchType.LAZY)
@JoinColumn(name="TEAM_ID") 
private Team team;
  • 이렇게 하면 Member를 조회할 때 , Team 테이블에 대한 Select Query가 나가지 않는다. 
  • 그리고 실제 team.getName() 처럼 Team을 실제로 사용하려들때가 되서야 Team 테이블에 대한 Select Query가 나간다.
  • 이게 어떻게 가능할까? 프록시 덕분에 가능하다. 

 

 

 

02. 프록시 기초

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

 

 

 

03. 프록시 특징

  • 실제 클래스를 상속받아서 만들어진다.
  • 실제 클래스와 겉 모양이 같다.
  • 사용하는 입장에선 진짜 객체인지 프록시 객체인지 구분하지 않고 사용하면 된다.
  • 프록시 객체는 실제 객체의 참조(target)를 보관
  • 프록시 객체를 호출하면 프록시 객체는 실제 객체의 메소드 호출

 

 

 

 

📍 프록시 객체의 초기화

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

 

 

  • 프록시 객체는 처음 사용할 때 한번만 초기화한다.
  • 프록시 객체를 초기화 할 때, 프록시 객체가 실제 엔티티로 바뀌는 것은 아니다.  초기화되면 프록시 객체를 통해서 실제 엔티티에 접근가능해지는 것이다. 
  • 프록시 객체는 원본 엔티티를 상속받는다. 따라서 타입 체크시 주의해야한다. (== 비교 실패. 대신 instance of 사용)
  • 영속성 컨텍스트에 찾는 엔티티가 이미 있으면 entityManager.getReference( )를 호출해도 실제 엔티티 반환한다.
  • 영속성 컨텍스트의 도움을 받을 수 없는 준영속 상태일 때, 프록시를 초기화하면 문제가 발생한다. (하이버네이트는 org.hibernate.LazyInitializationException 예외를 터트린다. )

 

 

04. 프록시 확인 

  • 프록시 인스턴스의 초기화 여부 확인
EntityManagerFactory.PersistenceUnitUtil.isLoaded(Object entity)
  • 프록시 클래스 확인 방법
entity.getClass( ).getName( )
  • 프록시 강제 초기화
org.hibernate.Hibernate.initialize(entity);

'이론 > JPA' 카테고리의 다른 글

[JPA] JPA를 사용해야 하는 이유  (0) 2021.09.26