[JPA] 영속성 전이(cascade)와 고아 객체
Updated:
1. 개요
cascade를 통해 엔티티를 영속성 컨텍스트에 올리거나 내릴 때, 관련된 엔티티도 함께 올리거나 내릴 수 있고, orphanRemoval을 통해 부모 엔티티와 관계가 끊어진 자식 엔티티를 자동으로 삭제할 수 있다. 이번에는 영속성 전이(cascade)와 고아객체에 대해 알아보도록 하자.
2. 영속성 전이(cascade)
cascade를 통해 특정 엔티티를 영속화 할 때 관련된 다른 엔티티도 함께 영속화 할 수 있다. 엔티티 연관관계 매핑과는 관련이 없고, 단순히 편리함을 제공하는 기능이다. 옵션으로는 ALL(모두 적용), PERSIST(영속), REMOVE(삭제), MERGE(병합), REFRESH, DETACH가 있는데, 주로 ALL, PERSIST, REMOVE를 사용한다.
2-1. 예제 코드
[Parent.java]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
@Entity
public class Parent {
@Id @GeneratedValue
private Long id;
private String name;
@OneToMany(mappedBy = "parent", cascade = CascadeType.ALL)
private List<Child> childList = new ArrayList<>();
public void addChild(Child child) {
childList.add(child);
child.setParent(this);
}
public Long getId() {
return id;
}
public void setId(Long id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public List<Child> getChildList() {
return childList;
}
public void setChildList(List<Child> childList) {
this.childList = childList;
}
}
Line 9 : cascade를 통해 관련된 엔티티들도 함께 persist
[Child.java]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
@Entity
public class Child {
@Id @GeneratedValue
private Long id;
private String name;
@ManyToOne
@JoinColumn(name = "parent_id")
private Parent parent;
public Long getId() {
return id;
}
public void setId(Long id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Parent getParent() {
return parent;
}
public void setParent(Parent parent) {
this.parent = parent;
}
}
[JpaMain.java]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
public class JpaMain {
public static void main(String[] args) {
EntityManagerFactory emf = Persistence.createEntityManagerFactory("hello");
EntityManager em = emf.createEntityManager();
EntityTransaction tx = em.getTransaction();
tx.begin();
try {
Child child1 = new Child();
Child child2 = new Child();
Parent parent = new Parent();
parent.addChild(child1);
parent.addChild(child2);
em.persist(parent);
tx.commit();
} catch (Exception e) {
tx.rollback();
e.printStackTrace();
} finally {
em.close();
}
emf.close();
}
}
2-2. 실행 결과
parent만 persist 했지만, cascade에 의해 child1, child2도 함께 insert 된 것을 확인할 수 있다.
3. 고아 객체(orphanRemoval)
orphanRemoval을 통해 부모 엔티티와 관계가 끊어진 자식 엔티티를 자동으로 삭제해줄 수 있다. 부모와 자식 엔티티의 라이프사이클이 동일하고, 특정 엔티티만 소유한 경우(참조하는 곳이 하나)에만 사용해야 한다.
3-1. 예제 코드
[Parent.java]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
@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<>();
public void addChild(Child child) {
childList.add(child);
child.setParent(this);
}
public Long getId() {
return id;
}
public void setId(Long id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public List<Child> getChildList() {
return childList;
}
public void setChildList(List<Child> childList) {
this.childList = childList;
}
}
Line 9 : orphanRemoval을 true로 설정하여 해당 컬렉션에서 빠진 child는 자동으로 삭제
[Child.java]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
@Entity
public class Child {
@Id @GeneratedValue
private Long id;
private String name;
@ManyToOne
@JoinColumn(name = "parent_id")
private Parent parent;
public Long getId() {
return id;
}
public void setId(Long id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Parent getParent() {
return parent;
}
public void setParent(Parent parent) {
this.parent = parent;
}
}
[JpaMain.java]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
public class JpaMain {
public static void main(String[] args) {
EntityManagerFactory emf = Persistence.createEntityManagerFactory("hello");
EntityManager em = emf.createEntityManager();
EntityTransaction tx = em.getTransaction();
tx.begin();
try {
Child child1 = new Child();
Child child2 = new Child();
Parent parent = new Parent();
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);
tx.commit();
} catch (Exception e) {
tx.rollback();
e.printStackTrace();
} finally {
em.close();
}
emf.close();
}
}
3-2. 실행 결과
컬렉션에서 제거된 엔티티에 대해 delete 쿼리가 나가는 것을 확인할 수 있다.
Leave a comment