- 특정 엔티티를 영속 상태로 만들 때 연관된 엔티티도 함께 영속 상태로 만들고 싶을 때
- 예) 부모 엔티티를 저장할 때 자식 엔티티도 함께 저장 (persist)
Parent와 Child 코드를 보자
@Entity
public class Parent {
@Id @GeneratedValue
private Long id;
private String name;
@OneToMany(mappedBy = "parent")
private List<Child> childList = new ArrayList<>();
public void addChild(Child child) {
child.setParent(this);
childList.add(child);
}
@Entity
public class Child {
@Id @GeneratedValue
private Long id;
private String name;
@ManyToOne
@JoinColumn(name = "PARENT_ID")
private Parent parent;
위 코드처럼 Parent와 Child가 양방향 연관관계로 매핑이 되어있다고 하자
Child child1 = new Child();
Child child2 = new Child();
Parent parent = new Parent();
parent.addChild(child1);
parent.addChild(child2);
em.persist(parent);
em.persist(child1);
em.persist(child2);
만약 Main코드를 위 코드처럼 작성을 하면 em.persist를 3번 호출해야 한다.
parent 중심으로 코드를 작성하고 있다.
parent가 persist 되면 child들은 자동으로 persist가 되길 원할 때 CASCADE를 사용한다.
@OneToMany(mappedBy = "parent", cascade = CascadeType.ALL)
private List<Child> childList = new ArrayList<>();
@OneToMany의 cascade 속성으로 CascadeType.ALL을 하면 된다.
그러면 em.persist(parent)만 해주어도 childList에 있는 child까지 persist 된다.
주의!
- 영속성 전이는 연관관계를 매핑하는 것과 아무 관련이 없다.
- 엔티티를 영속화할 때 연관된 엔티티도 함께 영속화하는 편리함을 제공할 뿐이다.
CASCADE의 종류
- ALL : 모두 적용
- PERSIST : 영속할 때만 적용
- REMOVE : 삭제할 때만 적용
- MERGE
- REFRESH
- DETACH
CASCADE를 사용하는 경우
- Parent와 Child의 Life 사이클이 유사할 때
- 단일 소유자 일 때
- Child의 연관관계가 오직 Parent와만 있는 경우
- Child가 Member나 다른 Entity와 연관관계가 있지 않은 경우
고아객체
- 고아 객체 제거 : 부모 엔티티와 연관관계가 끊어진 자식 엔티티를 자동 삭제하는 기능
- orphanRemoval = true
- Parent parent1 = em.find(Parent.Class, 1);
- parent1.getChild().remove(0); // 자식 엔티티를 컬렉션에서 제거
- 제거된 자식에 대한 DELETE SQL이 전송된다.
@Entity
public class Parent {
@Id @GeneratedValue
private Long id;
private String name;
@OneToMany(mappedBy = "parent", cascade = CascadeType.ALL, orphanRemoval = true)
private List<Child> childList = new ArrayList<>();
@OneToMany의 속성 orphanRemoval에 true로 넣어주면 된다.
고아 객체 주의
- 참조가 제거된 엔티티는 다른 곳에서 참조하지 않는 고아 객체로 보고 삭제하는 기능
- 참조하는 곳이 하나일 때 사용해야 한다.
- 특정 엔티티가 개인 소유할 때 사용
- @OneToOne, @OneToMany만 가능
- 부모를 제거하면 자식은 고아가 된다. 따라서 객체 제거 기능을 활성화하면, 부모를 제거할 때 자식도 함께 제거된다. 이것은 CascadeType.REMOVE처럼 동작한다.
- CascadeType.REMOVE vs orphanRemoval
- 공통점: Parent를 제거하면 Child도 제거된다.
- 차이점: Parent와 Child의 연관관계가 끊어졌을 때
- orphanRemoval만 Child가 제거된다.
- CascadeType.REMOVE에서 Child는 제거되지 않는다.
영속성 전이 + 고아 객체, 생명주기
- CascadeType.ALL + orphanRemoval=true
- 스스로 생명주기를 관리하는 엔티티는 em.persist()로 영속화, em.remove()로 제거
- 두 옵션을 모두 활성화 하면 부모 엔티티를 통해서 자식의 생명주기를 관리할 수 있다.
- 도메인 주도 설계의 Aggregate Root 개념을 구현할 때 유용하다.
'JPA' 카테고리의 다른 글
임베디드 타입 (0) | 2023.08.31 |
---|---|
기본값 타입 (0) | 2023.08.31 |
즉시 로딩과 지연 로딩 (0) | 2023.08.30 |
프록시와 연관관계 관리 (0) | 2023.08.30 |
MappedSuperclass (0) | 2023.08.30 |