Row was updated or deleted by another transaction (or unsaved-value mapping was

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 方法

可以强制进行从内存到数据库的同步。例:

  1. publicvoid testFlush(){
  2. Session session =HibernateUitl.getSessionFactory().getCurrentSession();
  3. session.beginTransaction();
  4. Teacher t =(Teacher) session.get(Teacher.class,3);
  5. t.setName("yyy");
  6. session.flush();//有flush会执行2次UPDAE,没有会只执行一次
  7. t.setName("yyyyy");
  8. session.getTransaction().commit();
  9. }

Flush方法是可以设置的,也就是 fulsh 什么时候执行是可以设置的,在session.beginTransaction 前设置 FlushMode

  1. 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 方法一起使用出现的时候,一定要注意使用的顺序,否知会出现异常。下面的代码就是正确的使用顺序:

  1. //因为user的主键生成策略采用的是uuid,所以调用完成save后,
  2. //只是将user纳入到了session的管理
  3. //不会发出insert语句,但是id已经生成,session中existsInDatebase状态为false
  4. session.save(user);
  5.  
  6. //flush后hibernate会清理缓存,会将user对象保存到数据库中
  7. //(强制更新数据库,结果但不一定,因为commit时候可能会出现事物回滚)
  8. //并且设置session中existsInDatebase的状态为true
  9. session.flush();
  10.  
  11. //将user对象从session中逐出,即session的EntityEntries属性中逐出
  12. session.evict(user);
  13.  
  14. //可以成功提交,因为hibernate在清理缓存时,
  15. //在session的insertions集合中无法找到user对象
  16. //所以就不会发出insert语句,也不会更新session中的existsInDatabase的状态
  17. tx.commit();

猜你喜欢

转载自aniyo.iteye.com/blog/2355364
Row