출처 : 자바 ORM 표준 JPA 프로그래밍 - 기본편
섹션 8. 프록시와 연관관계 관리
1. 프록시
1 - 1. 프록시 기초
- em.find() vs em.getReference()
- em.find(): 데이터베이스를 통해서 실제 엔티티 객체 조회
- em.getReference(): 데이터베이스 조회를 미루는 가짜(프록시) 엔티티 객체 조회
(실제 값이 사용되는 시점에 DB 쿼리를 함)
1 - 2. 프록시 특징
- 실제 클래스를 상속 받아서 만들어짐
- 실제 클래스와 겉 모양이 같다.
- 사용하는 입장에서는 진짜 객체인지 프록시 객체인지 구분하지 않고 사용하면 됨(이론상)
- 프록시 객체는 실제 객체의 참조(target)를 보관
- 프록시 객체를 호출하면 프록시 객체는 실제 객체의 메소드 호출
1 - 3. ⭐ 프록시의 특징 ⭐
- 프록시 객체는 처음 사용할 때 한 번만 초기화
- 프록시 객체를 초기화 할 때, 프록시 객체가 실제 엔티티로 바뀌는 것은 아님,
초기화되면 프록시 객체를 통해서 실제 엔티티에 접근 가능
- 프록시 객체는 원본 엔티티를 상속받음, 따라서 타입 체크시 주의해야함
(== 비교 실패, 대신 instance of 사용)
- 영속성 컨텍스트에 찾는 엔티티가 이미 있으면 em.getReference()를 호출해도 실제 엔티티 반환
- 영속성 컨텍스트의 도움을 받을 수 없는 준영속 상태일 때, 프록시를 초기화하면 문제 발생
(하이버네이트는 org.hibernate.LazyInitializationException 예외를 터트림)
2. 즉시 로딩과 지연 로딩
• 지연 로딩 LAZY을 사용해서 프록시로 조회
• 즉시 로딩 EAGER를 사용해서 함께 조회 (프록시 필요 X)
2 - 1. 🚨 프록시와 즉시로딩 주의 🚨
- 가급적 지연 로딩만 사용 (특히 실무에서)
- 즉시 로딩을 적용하면 예상하지 못한 SQL이 발생
- 즉시 로딩은 JPQL에서 N+1 문제를 일으킴
(최초 쿼리 : 1, 그거 때문에 추가 쿼리(N)가 나감)
- @ManyToOne, @OneToOne(X to One)은 기본이 즉시 로딩
-> LAZY로 설정
- @OneToMany, @ManyToMany는 기본이 지연 로딩
3. 지연 로딩 활용
3 - 1. 지연 로딩 활용 - 실무
- 모든 연관관계에 지연 로딩을 사용해라!
- 실무에서 즉시 로딩을 사용하지 마라!
- JPQL fetch 조인이나, 엔티티 그래프 기능을 사용해라!
(뒤에서 설명)
- 즉시 로딩은 상상하지 못한 쿼리가 나감
4. 영속성 전이: CASCADE
특정 엔티티를 영속 상태로 만들 때 연관된 엔티티도 함께 영속 상태로 만들도 싶을 때
(연관 관계 매핑 X)
예) 부모 엔티티를 저장할 때 자식 엔티티도 함께 저장
4 - 1. 🚨 영속성 전이: CASCADE - 주의!
- 영속성 전이는 연관관계를 매핑하는 것과 아무 관련이 없음
- 엔티티를 영속화할 때 연관된 엔티티도 함께 영속화하는 편리함을 제공할 뿐
4 - 2. CASCADE의 종류
- ALL: 모두 적용
- PERSIST: 영속
- REMOVE: 삭제
- MERGE: 병합
- REFRESH: REFRESH
- DETACH: DETACH
5. 고아 객체
고아 객체 제거: 부모 엔티티와 연관관계가 끊어진 자식 엔티티를 자동으로 삭제
orphanRemoval = true
5 - 1. 🚨 고아 객체 - 주의
- 참조가 제거된 엔티티는 다른 곳에서 참조하지 않는 고아 객체로 보고 삭제하는 기능
- 참조하는 곳이 하나일 때 사용해야함!
- 특정 엔티티가 개인 소유할 때 사용
- @OneToOne, @OneToMany만 가능
- 참고: 개념적으로 부모를 제거하면 자식은 고아가 됨.
따라서 고아 객체 제거 기능을 활성화 하면,
부모를 제거할 때 자식도 함께 제거됨.
이것은 CascadeType.REMOVE처럼 동작함.
6. 영속성 전이 + 고아 객체, 생명주기
섹션 9. 값 타입
1. 기본값 타입
1 - 1. JPA의 데이터 타입 분류
- 엔티티 타입
• @Entity로 정의하는 객체
• 데이터가 변해도 식별자로 지속해서 추적 가능
• 예) 회원 엔티티의 키나 나이 값을 변경해도 식별자로 인식 가능
- 값 타입
• int, Integer, String처럼 단순히 값으로 사용하는 자바 기본 타입이나 객체
• 식별자가 없고 값만 있으므로 변경시 추적 불가
• 예) 숫자 100을 200으로 변경하면 완전히 다른 값으로 대체
2. 값 타입 분류
- 기본값 타입
• 자바 기본 타입(int, double)
• 래퍼 클래스(Integer, Long)
• String
- 임베디드 타입(embedded type, 복합 값 타입)
- 컬렉션 값 타입(collection value type)
3. 임베디드 타입 (복합 값 타입)
- 새로운 값 타입을 직접 정의할 수 있음
- JPA는 임베디드 타입(embedded type)이라 함
- 주로 기본 값 타입을 모아서 만들어서 복합 값 타입이라고도 함
- int, String과 같은 값 타입
- 재사용
- 높은 응집도
3 - 1. 임베디드 타입과 테이블 매핑
테이블 설계와 객체 설계는 따로 할 수 있음.
- 임베디드 타입은 엔티티의 값일 뿐
- 임베디드 타입을 사용하기 전과 후에 매핑하는 테이블은 같음
- 객체와 테이블을 아주 세밀하게(find-grained) 매핑하는 것이 가능
- 잘 설계한 ORM 애플리케이션은 매핑한 테이블의 수보다 클래스의 수가 더 많음
예시
3 - 2. 값 타입과 불변 객체
4. 값 타입 컬렉션
- 값 타입을 하나 이상 저장할 때 사용
- @ElementCollection, @CollectionTable 사용
- 데이터베이스는 컬렉션을 같은 테이블에 저장할 수 없다.
- 컬렉션을 저장하기 위한 별도의 테이블이 필요함
4 - 1. 🚨 값 타입 컬렉션의 제약사항
- 값 타입 컬렉션에 변경 사항이 발생하면,
주인 엔티티와 연관된 모든 데이터를 삭제하고,
값 타입 컬렉션에 있는 현재 값을 모두 다시 저장한다.
4 - 2. 정리
📌 식별자가 필요하고, 지속해서 값을 추적, 변경해야 한다면 그것은 값 타입이 아닌 엔티티