1. 영속성 전이 : CASCADE
특정 엔티티를 영속 상태로 만들 때 연관된 엔티티도 함께 영속 상태로 만든다.
(1) 영속성 전이 : 저장
@Entity
@Getter @Setter
public class Parent {
@Id @GeneratedValue
private Long id;
private String name;
@OneToMany(mappedBy = "parent", cascade = CascadeType.PERSIST)
private List<Child> childList = new ArrayList<>();
public void addChild(Child child) {
childList.add(child);
child.setParent(this);
}
}
- cascade = CascadeType.ALL 선언
- 부모 엔티티를 저장할 때 자식 엔티티도 함께 저장된다.
@Entity
@Getter
@Setter
public class Child {
@Id @GeneratedValue
private Long id;
private String name;
@ManyToOne
@JoinColumn(name = "PARENT_ID")
private Parent parent;
}
//CASCADE(PERSIST)
Child child1 = new Child();
Child child2 = new Child();
Parent parent = new Parent();
parent.addChild(child1);
parent.addChild(child2);
//em.persist(child1);
//em.persist(child2);
em.persist(parent); //cascade 설정하면 child persist 생략가능하다.
- JPA에서 엔티티를 저장할 때 연관된 모든 엔티티는 영속 상태여야 한다.
- 부모 엔티티를 영속 상태로 만들고 자식 엔티티도 각각 영속 상태로 만든다.
- 영속성 전이(CASCADE)를 사용해서 부모만 영속 상태로 만들면 연관된 자식까지 한 번에 영속 상태로 만들 수 있다.
(2) 영속성 전이 : 삭제
@OneToMany(mappedBy = "parent", cascade = CascadeType.REMOVE)
//CASCADE(REMOVE)
Child child1 = em.find(Child.class, 1L);
Child child2 = em.find(Child.class, 2L);
Parent parent = em.find(Parent.class, 1L);
//em.remove(child1);
//em.remove(child2);
em.remove(parent); //cascade 설정하면 child remove 생략가능하다.
(3) CASCADE의 종류
- ALL : 모두 적용
- PERSIST : 영속
- REMOVE : 삭제
- MERGE : 병합
- REFERSH : refresh
- DETACH : detach
(4) 영속성 전이 주의점
- 영속성 전이는 연관관계를 매핑하는 것과 아무 관련이 없다.
- 엔티티를 영속화할 때 연관된 엔티티도 같이 영속화하는 편리함을 제공한다.
2. 고아 객체 제거
JPA는 부모 엔티티와 연관관계가 끊어진 자식 엔티티를 자동으로 삭제한다.
- 참조가 제거된 엔티티는 다른 곳에서 참조하지 않는 고아 객체로 보고 삭제하는 기능이다.
@Entity
@Getter @Setter
public class Parent {
@Id @GeneratedValue
private Long id;
private String name;
@OneToMany(mappedBy = "parent", orphanRemoval = true)
private List<Child> childList = new ArrayList<>();
public void addChild(Child child) {
childList.add(child);
child.setParent(this);
}
}
- orphanRemoval = true 옵션 설정
- 부모 엔티티의 컬렉션에서 자식 엔티티의 참조만 제거하면 자식 엔티티가 자동으로 삭제할 수 있다.
Parent findParent = em.find(Parent.class, parent.getId());
findParent.getChildList().remove(0);
- 위 코드를 통해서 컬렉션에서 첫 번째 자식을 제거하였다.
- orphanRemoval = true 옵션으로 인해 컬렉션에서 엔티티를 제거하면 데이터베이스의 데이터도 삭제된다.
- 고아 객체 제거 기능은 영속성 컨텍스트를 플러시 할 때 적용된다.
- 개념적으로 부모를 제거하면 자식은 고아가 된다. 따라서 고아 객체 제거 기능을 활성화하면, 부모를 제거할 때 자식도 함께 제거된다. => CascadeType.REMOVE 처럼 동작한다.
* 주의
- 참조하는 곳이 하나일 때 사용해야한다.
- 특정 엔티티가 개인 소유할 때 사용한다.
- @OneToOne, @OneToMany만 사용 가능하다.
3. 영속성 전이 + 고아 객체, 생명주기
- CascadeType.ALL + orphanRemoval = true를 동시에 사용하면 어떻게 될까?
- 두 옵션을 활성화하면 부모 엔티티를 통해서 자식의 생명주기를 관리할 수 있다.
(1) 저장 (Persist)
- 자식을 저장하려면 부모에 등록만 하면 된다. (CASCADE)
Parent parent = em.find(Parent.class, parentId);
parent.addChild(child);
(2) 제거 (Remove)
- 자식을 삭제하려면 부모에서 제거하면 된다. (orphanRemoval)
도메인 주도 설계(DDD)의 Aggregate Root 개념을 구현할 때 유용하다.
Parent parent = em.find(Parent.class, parentId);
parent.getChildren().remove(removeObject);
반응형
'JPA > JPA' 카테고리의 다른 글
[JPA] JPQL - 기본 문법 (0) | 2022.01.07 |
---|---|
[JPA] 값 타입 (0) | 2022.01.04 |
[JPA] 즉시 로딩과 지연 로딩 (0) | 2022.01.03 |
[JPA] 프록시 (Proxy) (0) | 2022.01.03 |
[JPA] @MappedSuperclass (0) | 2022.01.03 |
댓글