272

我想我误解了在关系中级联的含义@ManyToOne

案子:

public class User {

   @OneToMany(fetch = FetchType.EAGER)
   protected Set<Address> userAddresses;

}

public class Address {

   @ManyToOne(fetch = FetchType.LAZY, cascade = CascadeType.ALL)
   protected User addressOwner;

}

的含义是cascade = CascadeType.ALL什么?例如,如果我从数据库中删除某个地址,我添加的事实如何cascade = CascadeType.ALL影响我的数据(User我猜是)?

4

6 回答 6

429

的含义CascadeType.ALL是持久性将传播(级联)所有EntityManager操作(PERSIST, REMOVE, REFRESH, MERGE, DETACH)到相关实体。

在您的情况下,这似乎是一个坏主意,因为删除 anAddress会导致删除相关的User. 由于用户可以拥有多个地址,因此其他地址将成为孤立地址。但是,相反的情况(注释User)将有意义 - 如果地址仅属于单个用户,则在删除该用户时传播属于该用户的所有地址的删除是安全的。

顺便说一句:您可能希望向mappedBy="addressOwner"您添加一个属性,User以向持久性提供程序发出连接列应该在 ADDRESS 表中的信号。

于 2012-10-23T09:33:44.997 回答
80

你不应该使用CascadeType.ALLon@ManyToOne因为实体状态转换应该从父实体传播到子实体,而不是相反。

@ManyToOne侧面始终是子关联,因为它映射了基础外键列。

因此,您应该将关联CascadeType.ALL@ManyToOne关联移到@OneToMany一边,这也应该使用mappedBy属性,因为它是最有效的一对多表关系映射。

于 2017-08-10T12:14:15.433 回答
57

请参阅此处以获取 OpenJPA 文档中的示例。CascadeType.ALL意味着它将执行所有操作。

引用:

CascadeType.PERSIST:当持久化一个实体时,也持久化其字段中持有的实体。我们建议自由应用此级联规则,因为如果 EntityManager 在刷新期间发现一个引用新实体的字段,并且该字段不使用 CascadeType.PERSIST,则这是一个错误。

CascadeType.REMOVE:删除实体时,也会删除该字段中持有的实体。

CascadeType.REFRESH:刷新实体时,也刷新该字段中持有的实体。

CascadeType.MERGE:合并实体状态时,也合并该字段中持有的实体。

塞巴斯蒂安

于 2012-10-23T09:27:27.393 回答
21

来自 EJB3.0 规范

级联注释元素的使用可用于将操作的效果传播到关联实体。级联功能最常用于父子关系。

如果 X 是受管实体,则移除操作会导致它被移除。如果从 X 到这些其他实体的关系使用 cascade=REMOVE 或 cascade=ALL 注释元素值进行注释,则删除操作将级联到 X 引用的实体。

所以简而言之,定义的实体关系CascadeType.All将确保所有发生在父级上的持久性事件,例如持久化、刷新、合并和删除,都将传递给子级。定义其他CascadeType选项为开发人员提供了对实体关联如何处理持久性的更细粒度的控制。

例如,如果我有一个包含页面列表的对象 Book,并且我在此列表中添加了一个页面对象。如果@OneToMany定义 Book 和 Page 之间关联的注释标记为CascadeType.All,则持久化 Book 将导致 Page 也被持久化到数据库中。

于 2012-10-23T09:32:59.413 回答
13

在 JPA 2.0 中,如果您想删除从用户实体中删除的地址,您可以添加orphanRemoval=true(而不是CascadeType.REMOVE)到您的@OneToMany.

orphanRemoval=trueCascadeType.REMOVE之间的更多解释在这里

于 2015-08-06T08:55:17.383 回答
6

如果您只想删除分配给用户的地址而不影响用户实体类,您应该尝试以下操作:

@Entity
public class User {
   @OneToMany(mappedBy = "addressOwner", cascade = CascadeType.ALL)
   protected Set<Address> userAddresses = new HashSet<>();
}

@Entity 
public class Addresses {
   @ManyToOne(cascade = CascadeType.REFRESH) @JoinColumn(name = "user_id")
   protected User addressOwner;
}

这样你就不用担心在注解中使用 fetch 了。但请记住,在删除用户时,您还将删除连接到用户对象的地址。

于 2019-01-19T18:32:04.433 回答