org.hibernate.StaleObjectStateException: Row was updated or deleted by another transaction (or unsaved-value mapping was incorrect): [com.shkco.adsr3.cbm.vo.CustomerAccountInfo#74225]
at org.hibernate.persister.entity.AbstractEntityPersister.check(AbstractEntityPersister.java:2523) at org.hibernate.persister.entity.AbstractEntityPersister.update(AbstractEntityPersister.java:3242) at org.hibernate.persister.entity.AbstractEntityPersister.updateOrInsert(AbstractEntityPersister.java:3140) at org.hibernate.persister.entity.AbstractEntityPersister.update(AbstractEntityPersister.java:3470) at org.hibernate.action.internal.EntityUpdateAction.execute(EntityUpdateAction.java:140) at org.hibernate.engine.spi.ActionQueue.execute(ActionQueue.java:393) at org.hibernate.engine.spi.ActionQueue.executeActions(ActionQueue.java:385) at org.hibernate.engine.spi.ActionQueue.executeActions(ActionQueue.java:302) at org.hibernate.event.internal.AbstractFlushingEventListener.performExecutions(AbstractFlushingEventListener.java:339) at org.hibernate.event.internal.DefaultFlushEventListener.onFlush(DefaultFlushEventListener.java:52) at org.hibernate.internal.SessionImpl.flush(SessionImpl.java:1240) at com.shkco.adsr3.framework.dao.SessionWrapper.flush(SessionWrapper.java:209) at com.shkco.adsr3.framework.dao.AbstractDAO.flushAndClear(AbstractDAO.java:459) at com.shkco.adsr3.framework.service.AbstractBusinessService.flushAndClear(AbstractBusinessService.java:89)
使用Evict清除指定缓冲对象即可。
evict(custAcctInfo);
evict(custAcct);
一、Clear 方法
无论是Load 还是 Get 都会首先查找缓存(一级缓存) 如果没有,才会去数据库查找,调用Clear() 方法,可以强制清除Session缓存。
public void testClear(){ Session session = HibernateUitl.getSessionFactory().getCurrentSession(); session.beginTransaction(); Teacher t = (Teacher) session.get(Teacher.class, 3); System.out.println(t.getName()); session.clear();//这里不clear只会执行一次sql语句,有clear会执行2次 Teacher t2 = (Teacher) session.get(Teacher.class, 3); System.out.println(t2.getName()); session.getTransaction().commit(); }
二、Flush 方法
可以强制进行从内存到数据库的同步。例:
- publicvoid testFlush(){
- Session session =HibernateUitl.getSessionFactory().getCurrentSession();
- session.beginTransaction();
- Teacher t =(Teacher) session.get(Teacher.class,3);
- t.setName("yyy");
- session.flush();//有flush会执行2次UPDAE,没有会只执行一次
- t.setName("yyyyy");
- session.getTransaction().commit();
- }
Flush方法是可以设置的,也就是 fulsh 什么时候执行是可以设置的,在session.beginTransaction 前设置 FlushMode
- session.setFlushMode(FlushMode.Always|AUTO|COMMIT|NEVER|MANUAL);
这里 FlushMode 有 5 个值可选
Always: 任何代码都会 Flush
AUTO: 默认方式 – 自动
Commit: COMMIT 时
Never: 始终不
MANUAL: 手动方式
设置 FlushMode 有个好处是可以节省开销,比如默认 session 只做查询时,就可以不让他与数据库同步了。
三、Evict 方法
会把指定的缓冲对象进行清除,而 Clear 方法则是把缓冲区内的全部对象清除,但不包括操作中的对象。
注意事项: Flush 方法和 Evict 方法一起使用出现的问题
Flush 方法和 Evict 方法一起使用出现的时候,一定要注意使用的顺序,否知会出现异常。下面的代码就是正确的使用顺序:
- //因为user的主键生成策略采用的是uuid,所以调用完成save后,
- //只是将user纳入到了session的管理
- //不会发出insert语句,但是id已经生成,session中existsInDatebase状态为false
- session.save(user);
- //flush后hibernate会清理缓存,会将user对象保存到数据库中
- //(强制更新数据库,结果但不一定,因为commit时候可能会出现事物回滚)
- //并且设置session中existsInDatebase的状态为true
- session.flush();
- //将user对象从session中逐出,即session的EntityEntries属性中逐出
- session.evict(user);
- //可以成功提交,因为hibernate在清理缓存时,
- //在session的insertions集合中无法找到user对象
- //所以就不会发出insert语句,也不会更新session中的existsInDatabase的状态
- tx.commit();