河堤小竹
逝者如斯夫,不舍昼夜!

Hibernate中evict方法和clear方法说明

  先创建一个对象,然后调用session.save方法,然后调用evict方法把该对象清除出缓存,最后提交事务。结果报错:

Exception in thread "main" org.hibernate.AssertionFailure: possible nonthreadsafe access to session。如果

用的不是evict方法,而是clear方法来清session缓存的话,就不会报错。

1.session.evict(obj);

    明显是清除缓冲中指定的对象。

2.session.clear();

    没带参数,是清除掉缓冲区中所有的对象,但不包含当前对象。

3.hibernate执行的顺序如下

  (1)生成一个事务的对象,并标记当前的session处于事务状态(此时并未启动数据库级事务)

  (2)应用使用session.save保存对象,这个时候session将这个对象放入EntityEntries,用来标记对象以及和当前的会话建立了关联,由于应用对对象做个保存的操作,

session还要在insertions中登记应用的这个插入行为(行为包括:对象引用,对象id,session,持久化处理类)。

  (3)session.evict方法将对象从session会话中清除,这时session会从EntityEntries中将这个对象移出。

  (4)事务提交需要将所有缓存flush入数据库,Session启动一个事务,并按照insert,update,……,delete的顺序提交所有值钱登记的操作(注意:所有insert执行完毕后才会执行

update,这里的特殊处理也可能会将程序搞坏,如果需要控制操作的执行顺序,要善于使用flush),现在对象不再EntityEntries中,但在执行insert的行为时只需要访问insertions就够了,

所以此时不会有任何异常。异常出现在插入后通知session该对象已经插入完毕这个步骤上,这个步骤中需要将EntityEntries中对象的existsInDatabase标志置为true,由于对象并不存在于

EntityEntries中,此时Hibernate就认为insertions和EntityEntries可能因为线程安全问题产生了不同步,于是抛出一个net.sf.hibernate.AssertionFailure,导致程序终止。

  一般错误的认为session.save会立即执行,而将对象过早的从session中清除,造成session的insertions和EntityEntries中内容的不同步。所有我们在做此类操作时一定要明白

Hibernate什么时候会将数据flush入数据库,在未flush之前不要将已经进行过save的对象从session中清除。

  解决办法就是在save后,执行session.flush();。


 

posted on 2016-07-28 16:04  河堤小竹  阅读(592)  评论(0编辑  收藏  举报