代码改变世界

hibernate 并发机制与持久化对象

2017-03-03 09:39  Eric-Yuan  阅读(712)  评论(0编辑  收藏  举报

hibernate的session属于非线程安全,所以在用户量多的时候可能造成多个用户同时访问同一session,可能会引发第一类丢失更新,脏读,幻读,不可重复读,第二类丢失更新一系列的问题。

 

解决并发机制的方案:

  解决方案:设置事务隔离级别。  
    Serializable:串行化。隔离级别最高  
    Repeatable Read:可重复读  
    Read Committed:已提交数据读  
    Read Uncommitted:未提交数据读。隔离级别最差  

Hibernate 通过为 Hibernate 映射文件指定 hibernate.connection.isolation 属性来设置事务的隔离级别。例:

hibernate.connection.isolation = 4

 

在 Hibernate 中设置隔离级别

JDBC 数据库连接使用数据库系统默认的隔离级别在 Hibernate 的配置文件中可以显式的设置隔离级别每一个隔离级别都对应一个整数:  

隔离级别 对应的整数表示  
READ UNCOMMITED 1
READ COMMITED 2
REPEATABLE READ 4
SERIALIZEABLE 8


    设置锁:乐观锁和悲观锁。  
    乐观锁:使用版本号或时间戳来检测更新丢失,在的映射中设置 optimistic-lock=”all”可以在没有版本或者时间戳属性映射的情况下实现 版本检查,此时Hibernate将比较一行记录的每个字段的状态 行级悲观锁:Hibernate总是使用数据库的锁定机制,从不在内存中锁定对象!只要为JDBC连接指定一下隔 离级别,然后让数据库去搞定一切就够了。类LockMode 定义了Hibernate所需的不同的锁定级别:LockMode.UPGRADE,LockMode.UPGRADE_NOWAIT,LockMode.READ;

 

update和saveOrUpdate的区别?

  update()和saveOrUpdate()是用来对跨Session的PO进行状态管理的。  
  update()方法操作的对象必须是持久化了的对象。也就是说,如果此对象在数据库中不存在的话,就不能使用update()方法。  
  saveOrUpdate()方法操作的对象既可以使持久化了的,也可以使没有持久化的对象。如果是持久化了的对象调用saveOrUpdate()则会  更新数据库中的对象;如果是未持久化的对象使用此方法,则save到数据库中。

hibernate的三种状态之间如何转换  
  当对象由瞬时状态(Transient)一save()时,就变成了持久化状态;  
  当我们在Session里存储对象的时候,实际是在Session的Map里存了一份,  也就是它的缓存里放了一份,然后,又到数据库里存了一份,在缓存里这一份叫持久对象(Persistent)。  Session 一 Close()了,它的缓存也都关闭了,整个Session也就失效了,这个时候,这个对象变成了游离状态(Detached),但数据库中还是存在的。  
  当游离状态(Detached)update()时,又变为了持久状态(Persistent)。  
  当持久状态(Persistent)delete()时,又变为了瞬时状态(Transient), 此时,数据库中没有与之对应的记录。