准备从今天开始收集JavaEE中相关的Exception,并给出解决方案,希望大家有遇到各种类型的异常也跟上来!!!!
1.org.hibernate.NonUniqueObjectException: a different object with the same identifier value was already associated with the session: [com.test.sshprj.entity.CstCustomer#111]
发生这种异常的原因是在执行修改操作的时候,session管理两个具有相同标识符的持久对象造成的。下面代码详细说明:
@Test @Rollback(false) //使用spring的Test环境做测试,所以事务由spring自动处理 public void testAdd() { SysUsers newUser=new SysUsers(); newUser.setId(3L); newUser.setUserName("jason"); newUser.setPassword("jason"); this.update(newUser); } private void update(SysUsers newUser){ Session session=this.userdao.getSessionFactory().getCurrentSession(); SysUsers user=(SysUsers) session.get(SysUsers.class,3L); newUser.setEmail(user.getEmail());//把未修改的属性设置到脱管对象中 newUser.setLastLoginTime(user.getLastLoginTime()); session.update(newUser); //修改脱管对象newUser 抛出异常NonUniqueObjectException }
解决方案1:修改持久对象,把脱管对象中修改过的属性复制到持久对象中
public void testAdd() { SysUsers newUser=new SysUsers(); newUser.setId(3L); newUser.setUserName("jason"); newUser.setPassword("jason"); this.update(newUser); } private void update(SysUsers newUser){ Session session=this.userdao.getSessionFactory().getCurrentSession(); SysUsers user=(SysUsers) session.get(SysUsers.class,3L); user.setUserName(newUser.getUserName()); user.setPassword(newUser.getPassword()); session.update(user); }
解决方案2:使用session.merge(newUser)
merge的内部,是把newUser对象的属性拷贝到user对象中,然后再修改user对象
private void update(SysUsers newUser){ Session session=this.userdao.getSessionFactory().getCurrentSession(); SysUsers user=(SysUsers) session.get(SysUsers.class,3L); newUser.setEmail(user.getEmail());//把未修改的属性设置到脱管对象中 newUser.setLastLoginTime(user.getLastLoginTime()); session.merge(newUser); //修改脱管对象newUser 抛出异常NonUniqueObjectException }
解决方案3:使用session的evict方法,evict会把持久对象从一级缓存中删除掉,这样再修改newUser对象将不会出现问题
private void update(SysUsers newUser){ Session session=this.userdao.getSessionFactory().getCurrentSession(); SysUsers user=(SysUsers) session.get(SysUsers.class,3L); newUser.setEmail(user.getEmail());//把未修改的属性设置到脱管对象中 newUser.setLastLoginTime(user.getLastLoginTime()); session.evict(user); session.update(newUser); }