【发布时间】:2011-04-02 23:08:48
【问题描述】:
当我使用 DAO.update(userbean) 时,session.SaveOrUpdate(e);throw 异常:具有相同标识符值的不同对象已与会话关联
函数如下:
public E save(E e) {
Session session = null;
try {
session = sessionFactory.openSession();
log.debug("session="+session.hashCode()+" save "+e);
session.SaveOrUpdate(e); //here throws exception
session.flush();
}
catch (Exception e1) {
log.err("Cannot open hibernate session "+ e1.getMessage()+" cause : "+e1.getCause());
e1.printStackTrace();
}
finally { if ( session != null ) session.close(); session = null;}
return e ;
}
userbean 是类 UserBean 的实例
public class UserBean{
private List<GroupBean> groups = new ArrayList<GroupBean> ();
private List<RoleBean> roles = new ArrayList<RoleBean> ();
}
public class GroupBean{
private List<RoleBean> roles = new ArrayList<RoleBean> ();
}
每个 groupbean 都有一个角色列表,这些角色不会更改。
在数据库中,组和角色是多对多的映射,
例如,
我们有一个 groupbean#1,它的角色:rolebean#1,rolebean#2;
groupbean#2,哪些角色是rolebean#1。
现在我创建了一个新的 userbean#1,它的组是 groupbean#1 如果我想将rolebean#1添加到userbean#1,它会像标题描述一样抛出异常
查看server.log,发现当我使用DAO.save时,saveOrUpdate的顺序是:
userbean#1
|---|-----------***userbean.groups
| | groupbean#1
| | groupbean.roles
| | rolebean#1 # save relebean#1 the first time
| | ---done rolebean#1
| | ------done all rolebeans of group.roles
| | ---done groupbean#1
| |-----------done all groupbeans of userbean.groups
|---|-----------***userbean.roles
| rolebean#1 # save rolebean#1 the second time, and throws exception here!
| ----done rolebean#1
| .....
|-----------done all rolebeans of userbean.roles
异常的原因是rolebean#1在一个会话中被保存了两次,它们的身份是一样的。
在函数save(E e)中,如果我使用
session.merge(e);
替换
session.SaveOrUpdate(e);
不会抛出异常,但 rolebean#1 没有关联到 userbean#1
任何人都可以对此提出一些建议吗?
【问题讨论】: