在使用hibernate+spring做级联删除的时候,有时候我们会遇到这个异常:
deleted object would be re-saved by cascade (remove deleted object from associations)
比如说,论坛模块中,一个帖子对应多个一级评论,每个一级评论又对应多个二级评论。
这个时候,帖子和一级评论是双向一对多,一级评论和二级评论是双向的一对多关系。
我们在设置表关系的时候,一般是把一对多的一端的set设置inverse="true" cascade="all" 这两个属性。也就是说,控制权交给了多的一端,然后级联操作是所有操作都级联。
此时当我们要删除一级评论的时候,就可能出现这个异常。
因为,一级评论既是帖子的多端,也是二级评论的一端。按照道理,应该只是把一级评论及其对应的二级评论的记录都删除就对了。而且我们设置的也完全合理。但是还是会出现这个异常。
我们在使用spring+hibernate的时候,一般是使用spring来管理事务,如果我们直接使用hibernate来进行删除操作,并且事务都是手动开启手动提交的,这时有可能就不会出现这个异常了。但是,既然使用了spring,不用spring的事务就有点浪费了。
Session session=this.getHibernateTemplate().getSessionFactory().openSession(); session.beginTransaction(); session.delete(topicFirstDis); session.getTransaction().commit(); session.close();
具体解决办法:
方法一:(已测试,可用)
在调用删除方法的前面加this.getHibernateTemplate().clear();来清理缓存。有可能的原因是因为这个session查询到的这个一级评论是某一个已经查询出来的帖子的子记录。所以会提示我们从级联中移除这个要删除的对象。
代码如下:
this.getHibernateTemplate().clear(); this.getHibernateTemplate().delete(topicFirstDis);
这样有可能就可以进行删除了。
方法二:(已测试,可用)
把这个一级评论和帖子的关联关系去掉后,再进行删除操作
topic.getTopicFirstDisSet().remove(topicFirstDis); topicFirstDis.setTopic(null); boolean result=topicDao.deleteFirstDiscuss(topicFirstDis);
在帖子的set集合中把这个要删除的一级评论remove出去,同时把该一级评论的父对象属性设为null。然后再调用这个一级评论的删除方法。
方法三:(已测试,可用)
在直接使用hibernate操作的时候,也可能出现这个异常,这个时候,我们也需要按照上面的方法二的方式把帖子对象中的要删除的一级评论先remove出去,然后再把这个一级评论的父对象的属性设为null,就可以了。
当然也可以试试下面的代码,原理其实都是一样的。
Session session=this.getHibernateTemplate().getSessionFactory().openSession(); session.beginTransaction(); //把一级评论从帖子中移除 Topic topic=(Topic) session.load(Topic.class, topicFirstDis.getTopic().getTopicid()); topic.getTopicFirstDisSet().remove(topicFirstDis); session.delete(topicFirstDis); session.getTransaction().commit(); session.close();
方法四:(未测试,摘自网络)
删除Set方的cascade:(缺点:子对象和父对象不能进行级联更新,没意义)
方法五:(未测试,摘自网络)
在many-to-one方增加cascade 但值不能是none (缺点:子对象也级联到父对象,难得删除一个学生要把老师也删除?)
方法六:(未测试,摘自网络)
直接把子对象关联的父对象setNull,然后保存子对象,然后删除子对象,具体看代码
thisMany.setOne(null); Service.update(thisMany); Service.delete(thisMany);