본문 바로가기

Spring/JPA & Hibernate

영속성 전이 CASCADE

반응형

영속성 전이 CASCADE

특정 엔티티를 영속 상태로 만들 때 연관된 엔티티도 함께 영속 상태로 만들도 싶을 때 사용하는 기능

 


@Entity
public class Parent {

	@Id
	@GeneratedValue
	private Long id;

	private String name;

	@OneToMany( mappedBy = "parent" )
	private List<Child> childList = new ArrayList<Child>();
}

@Entity
public class Child {

	@Id
	@GeneratedValue
	private Long id;

	private String name;

	@ManyToOne
	@JoinColumn( name = "parent_id" )
	private Parent parent;
}

Parent parent = new Parent();
Child child1 = new Child();
Child child2 = new Child();

parent.addChild( child1 );
parent.addChild( child2 );

em.persist( child1 );
em.persist( child2 );
em.persist( parent );

em.flush();
em.clear();

위와 같이 parent 를 persist() 하기 위해 child 들을 모두 persist() 해야 한다.

이러한 번거로움을 없애기 위해 cascade = CascadeType.ALL 추가하면 연관된 엔티티(child)도 함께 persist 된다.


@Entity
public class Parent {

	@Id
	@GeneratedValue
	private Long id;

	private String name;

	@OneToMany( mappedBy = "parent", cascade = CascadeType.ALL )
	private List<Child> childList = new ArrayList<Child>();
}

 

CASCADE 를 ALL/PERSIST로 설정할 경우 같이 부모 엔티티를 저장할 때 자식 엔티티도 함께 저장하게 된다.

CASCADE의 종류

  • ALL: 모두 적용하겠다
  • PERSIST: 영속 => 저장할때만 사용하겠다.
  • REMOVE: 삭제
  • MERGE: 병합
  • REFRESH: REFRESH 
  • DETACH: DETACH

영속성 전이: CASCADE - 주의!

  • 영속성 전이는 연관관계를 매핑하는 것과 아무 관련이 없음
  • 엔티티를 영속화할 때 연관된 엔티티도 함께 영속화하는 편리함을 제공할 뿐
  • 단일 소유자. 즉 소유한 엔티티가 하나일 때 (child와 연관 관계가 있는(를 소유한) 엔티티가 parent 뿐일 때)만 사용.
    하나의 테이블에 완전히 종속적일때 (두 엔티티의 라이프사이클(CRUD)이 동일할 때) 사용

    ✔︎ 다른 엔티티도 child와 연관관계가 있을 경우에는 쓰면 안된다. 다 따로 관리하는 것이 맞음.

고아 객체

  • 고아 객체 제거: 부모 엔티티와 연관관계가 끊어진 자식 엔티티를 자동으로 삭제
  • orphanRemoval = true  // 자식 엔티티를 컬렉션에서 제거 

@OneToMany( mappedBy = "parent", cascade = CascadeType.ALL, orphanRemoval = true )
private List<Child> childList = new ArrayList<Child>();

Parent parent = new Parent();
Child child1 = new Child();
Child child2 = new Child();

parent.addChild( child1 );
parent.addChild( child2 );

em.persist( parent );

em.flush();
em.clear();

Parent findParent = em.find( Parent.class, parent.getId() );
findParent.getChildList().remove( 0 );

DELETE FROM CHILD WHERE ID=?

실행 결과에서 볼 수 있듯이, orphanRemoval을 true로 설정할 경우, PARENT 삭제 시 CHILD 역시 DELETE 쿼리가 발생한다.

 

고아 객체 주의

(cascade all/remove 처럼 참조가 제거된 엔티티는 다른 곳에서 참조하지 않는 고아 객체로 보고 삭제하는 기능 )

  • 참조하는 곳이 하나일 때 사용해야함! 
  • 특정 엔티티가 개인 소유할 때 사용
  • @OneToOne, @OneToMany만 가능
  • 참고: 개념적으로 부모를 제거하면 자식은 고아가 된다. 따라서 고아 객체 제거 기능을 활성화 하면, 부모를 제거할 때 자식도 함께 제거된다. 이것은 CascadeType.REMOVE 처럼 동작한다.

Parent parent = new Parent();
Child child1 = new Child();
Child child2 = new Child();

parent.addChild( child1 );
parent.addChild( child2 );

em.persist( parent );

em.flush();
em.clear();

Parent findParent = em.find( Parent.class, parent.getId() );
em.remove( findParent );

모든 child delete 쿼리 발생 후에 parent 제거

CascadeType.ALL + orphanRemovel=true 의 의미

  • 스스로 생명주기를 관리하는 엔티티는 em.persist()로 영속화, em.remove()로 제거한다.
  • 만약, 두 옵션을 모두 활성화 하면 부모 엔티티를 통해서 자식의 생명 주기를 관리할 수 있다
    즉, DAO나 Repository가 없어도 된다.
  • 이는 도메인 주도 설계(DDD)의 Aggregate Root개념을 구현할 때 유용

Aggregate Root 참고

https://medium.com/@SlackBeck/%EC%95%A0%EA%B7%B8%EB%A6%AC%EA%B2%8C%EC%9E%87-%ED%95%98%EB%82%98%EC%97%90-%EB%A6%AC%ED%8C%8C%EC%A7%80%ED%86%A0%EB%A6%AC-%ED%95%98%EB%82%98-f97a69662f63

 

 

 

반응형

'Spring > JPA & Hibernate' 카테고리의 다른 글

Transaction  (0) 2020.12.15
JPA 데이터 타입  (0) 2020.01.15
proxy  (0) 2020.01.10
@MappedSuperclass  (0) 2020.01.09
연관관계 맵핑  (0) 2019.12.19