出错:a different object with the same identifier value was already associated with the session
public boolean update(BusLineinfo busLineinfo) {
boolean isUpdate = false;
try{
BusLineinfo line = busLineinfoDao.queryByLineNameAndfromorto(busLineinfo.getLinename(), busLineinfo.getFromorto());//A、得出来的line是持久化的对象
if(line==null || line.getLineid().equals(busLineinfo.getLineid())){
isUpdate = busLineinfoDao.update(busLineinfo);//B、调用的是下面的update方法,得出来的也是持久化的对象
isUpdate = true;
}else{
isUpdate = false;
}
}catch(RuntimeException re){
log.error("BusLineinfoServiceImpl update is faield", re);
re.printStackTrace();
}
return isUpdate;
}
public boolean update(BusLineinfo busLineinfo) {
try {
getHibernateTemplate().update(busLineinfo);//C、想要改变这个错误,可以将其中的update方法改为merge方法
return true;
} catch (RuntimeException re) {
log.error("BusLineinfoDaoImpl update is failed", re);
re.printStackTrace();
}
return false;
}
分析:按照以上的写法就会抛出上面所说的错误,我们回过头来分析一下,出现这个问题的原因。
首先,我们用的是spring+hibernate+struts2,我们的事务是加在了第一个方法上。第一个方法,在A和B两个地方得出了两个持久化对象,并且这两个持久化的对象的主键是相同的。
另外,持久化的对象都会放到缓存里面,现在就出现了一个缓存中出现了两个主键相同的对象(这是不被允许的,就好比数据库中不允许主键相同是一样的)。
解法方法:将第二个方法的update方法改为merge方法。
附merge说明:
merge(): 他说的就是如果没有该条记录,则会被创建; 如果有该条记录,但内容有变,则会被修改; 相当于save or update. merge 和saveOrUpdate不一样,当session中某持久化对象有id相同的两个纪录时,必须用merge,merge会在保存之前来合并记录,不然会报错。合并记录后的动作和 saveOrUpdate一样。 |