JPA 一对多级联更新时,不能级联删除.
悬赏:30 发布时间:2008-07-22 提问人:ltsiphon333 (初级程序员)
级联新增,删除都ok;
级联更新时,修改从表内容的update,新增也ok,就是删除不ok.
以下贴出代码部分,望大家能够指点.
[code]
public class Account implements java.io.Serializable
{
private String id;
private String name;
private String enabled;
private Set<Acct2group> acct2groups = new HashSet<Acct2group>(0);
@OneToMany(cascade = CascadeType.ALL, mappedBy = "account")
public Set<Acct2group> getAcct2groups()
{
return this.acct2groups;
}
[code]
将Account类查出后,
[code]
Account account = dao.findById(1);
Set set = account.getAcct2groups();//此时set结果集为4
set.clear();
em.merge(account);
[code]
事务是spring控管的,做别的持久层动作都没问题.
就是不能将remove掉的4个Acct2group对象,重DB中删除.
将set.clear换成set.iterator().hasNext()迭代remove也不可以.
大家伙帮忙看看啊.
问题补充:
kamhung 你好,
我用的是hibernate,org.hibernate.annotations.CascadeType没有DELETE_OPTION这个属性啊.只有DELETE和DELETE_ORPHAN.我两个都试了一下,都没有达到删除孤子的效果啊,还望再次帮助.
级联更新时,修改从表内容的update,新增也ok,就是删除不ok.
以下贴出代码部分,望大家能够指点.
[code]
public class Account implements java.io.Serializable
{
private String id;
private String name;
private String enabled;
private Set<Acct2group> acct2groups = new HashSet<Acct2group>(0);
@OneToMany(cascade = CascadeType.ALL, mappedBy = "account")
public Set<Acct2group> getAcct2groups()
{
return this.acct2groups;
}
[code]
将Account类查出后,
[code]
Account account = dao.findById(1);
Set set = account.getAcct2groups();//此时set结果集为4
set.clear();
em.merge(account);
[code]
事务是spring控管的,做别的持久层动作都没问题.
就是不能将remove掉的4个Acct2group对象,重DB中删除.
将set.clear换成set.iterator().hasNext()迭代remove也不可以.
大家伙帮忙看看啊.
问题补充:
kamhung 你好,
我用的是hibernate,org.hibernate.annotations.CascadeType没有DELETE_OPTION这个属性啊.只有DELETE和DELETE_ORPHAN.我两个都试了一下,都没有达到删除孤子的效果啊,还望再次帮助.
采纳的答案
2008-07-23 kamhung (中级程序员)
JPA规范是不支持级联删除, 不过一般实现方都会提供扩展功能来支持。
如果你用的是toplink实现包的话, 你就要实现一个DescriptorCustomizer 接口
public class AccountDescriptorCustomizer implements DescriptorCustomizer {
public void customize(ClassDescriptor descriptor) throws Exception {
OneToManyMapping mapping = (OneToManyMapping) descriptor.getMappingForAttributeName("acct2groups");
if (mapping != null) mapping.setIsPrivateOwned(true);
}
}
然后在persistence.xml 中properties节点下加入:
<property name="toplink.descriptor.customizer.包名.Account" value="包名.AccountDescriptorCustomizer"/>
最后Acct2group 中 @ManyToOne 属性cascade将CascadeType.ALL 改成 {CascadeType.REFRESH, CascadeType.MERGE, CascadeType.MERGE}.
如果是hibernate的实现的话, 在@OneToMany
旁边加上: @org.hibernate.annotations.Cascade(org.hibernate.annotations.CascadeType.DELETE_OPTION)
如果你用的是toplink实现包的话, 你就要实现一个DescriptorCustomizer 接口
public class AccountDescriptorCustomizer implements DescriptorCustomizer {
public void customize(ClassDescriptor descriptor) throws Exception {
OneToManyMapping mapping = (OneToManyMapping) descriptor.getMappingForAttributeName("acct2groups");
if (mapping != null) mapping.setIsPrivateOwned(true);
}
}
然后在persistence.xml 中properties节点下加入:
<property name="toplink.descriptor.customizer.包名.Account" value="包名.AccountDescriptorCustomizer"/>
最后Acct2group 中 @ManyToOne 属性cascade将CascadeType.ALL 改成 {CascadeType.REFRESH, CascadeType.MERGE, CascadeType.MERGE}.
如果是hibernate的实现的话, 在@OneToMany
旁边加上: @org.hibernate.annotations.Cascade(org.hibernate.annotations.CascadeType.DELETE_OPTION)
提问者对于答案的评价:
多谢了.
其他回答
你描述的这种需求不属于级联删除,属于hibernate中的级联删除孤子,在jpa里面没有。
只能手动删除
只能手动删除
nihongye (中级程序员) 2008-07-22
可以这么做,
account.setAcct2grooups(null);
em.merge(account);
account.setAcct2grooups(null);
em.merge(account);
bianqioujin (初级程序员) 2008-07-23
可以这么做,
account.setAcct2grooups(null);
em.merge(account);
account.setAcct2grooups(null);
em.merge(account);
bianqioujin (初级程序员) 2008-07-23




