JPA级联关系@OneToOne、@OneToMany、@ManyToOne和@ManyToMany可以设置属性CascadeType,默认值为空。对于JPA2.0来说,CascadeType包含以下几种:
ALL, (/*Cascade all operations/)
PERSIST, (/* Cascade persist operation /)
MERGE, (/* Cascade merge operation /)
REMOVE,(/* Cascade remove operation /)
REFRESH,(/* Cascade refresh operation /)
DETACH(Cascade detach operation, @since Java Persistence 2.0).
简单介绍这些类型的用法之前,我们先建立两个级联实例。
@Entity
@Table(name = "policy")
public class EdgePolicy{
@Id
private int id;
private String name;
@OneToMany(mappedBy = "plan", fetch = FetchType.LAZY)
private List<PolicyHostMap> maps;
}
@Entity
@Table(name = "hostPolicyMap")
public class PolicyHostMap {
@Id
private int hostid;
private int policyid;
private int status;
@ManyToOne
@PrimaryKeyJoinColumn(name="policyid", referencedColumnName="id")
private EdgePolicy plan;
}
- CascadeType.DETACH
这种级联类型对应的是entityManager.detach(entity)方法。我们知道detach方法的作用是Remove the given entity from the persistence context。举例说明:
首先,我们需要对EdgePolicy实例中的maps添加CascadeType.DETACH,然后编写测试用例。
@OneToMany(mappedBy = "plan", cascade = {CascadeType.DETACH}, fetch = FetchType.LAZY)
private List<PolicyHostMap> maps;
entityManager.getTransaction().begin();
EdgePolicy policy = entityManager.findById(1);
policy.setName("new plan name");
List<PolicyHostMap> hostMaps = policy.getMaps();
for(PolicyHostMap hostMap : hostMaps){
hostMap.setEnablestatus(9);
}
entityManager.detach(policy);
entityManager.getTransaction().commit();
我们在测试用例中,虽然对实例policy重新赋值,但是在提交事务之前,又detach了policy实例,所以将不会update数据库表policy的name。那么,关联的hostPolicyMap会怎样呢?
检查测试结果,我们会发现如果没有DETACH级联关系,数据库表hostPolicyMap的status会更新为9。反之,将不被更新。
2. CascadeType.REFRESH
CascadeType.REFRESH对应的是entityManager.refresh(entity)。如果设置这种级联关系,当refresh(policy)时,List也将会执行refresh操作。举例说明,这次我们加在PolicyHostMap实例。
@ManyToOne(cascade = CascadeType.REFRESH)
@PrimaryKeyJoinColumn(name="policyid", referencedColumnName="id")
private EdgePolicy plan;
policy.setName("new plan name");
List<PolicyHostMap> hostMaps = policy.getMaps();
注意:由于policy懒加载maps,所以当我们执行 policy.getMaps()时,会refresh关联表PolicyHostMap,由于PolicyHostMap上对EdgePolicy有refresh标签,所以policy也会做refresh,会重新读取数据库并重置policy对象,使我们对name的重新赋值无效。
3. PERSIST, MERGE, REMOVE标签的作用就是分别对应entityManager的persist(级联保存)、merge(级联更新)和remove(级联删除)。
4. CascadeType.ALL要慎用,它包含所有的级联关系。如果我们要指定多种级联关系,可以把需要的CascadeType都列上,尽量不要使用ALL。例如我们想指定policy对policyhostmap有detach和refresh的级联关系,代码如下:
@OneToMany(mappedBy = "plan", cascade = {CascadeType.DETACH, CascadeType.REFRESH}, fetch = FetchType.LAZY)
private List<PolicyHostMap> maps;
总之,在使用jpa级联关系的时候,如果我们要在实例A上对实例B加级联关系,我们应该确定当对实例A进行detach、refresh、remove等操作时,B需不需要做相应的变化。