JPA

영속성 전이 : CASCADE, 고아객체

salmon16 2023. 8. 31. 00:58
  • 특정 엔티티를 영속 상태로 만들 때 연관된 엔티티도 함께 영속 상태로 만들고 싶을 때
  • 예) 부모 엔티티를 저장할 때 자식 엔티티도 함께 저장 (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.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