`

悲观锁 乐观锁

 
阅读更多

悲观锁( Pessimistic Locking
悲观锁,正如其名,他是对数据库而言的,数据库悲观了,他感觉每一个对他操作的程序都有可能产生并发。它指的是对数据被外界(包括本系统当前的其他事务,以及来自外部系统的事务处理)修改持保守态度,因此,在整个数据处理过程中,将数据处于锁定状态。悲观锁的实现,往往依靠数据库提供的锁机制(也只有数据库层提供的锁机制才能真正保证数据访问的排他性,否则,即使在本系统中实现了加锁机制,也无法保证外部系统不会修改数据)。
一个典型的倚赖数据库的悲观锁调用:

[sql] view plaincopy
 
  1. select * from account wherename=”Erica” forupdate  
[sql] view plain copy
 
  1. select * from account wherename=”Erica” forupdate  

这条 sql语句锁定了 account表中所有符合检索条件(name=Erica)的记录。本次事务提交之前(事务提交时会释放事务过程中的锁),外界无法修改这些记录。Hibernate的悲观锁,也是基于数据库的锁机制实现。
下面的代码实现了对查询记录的加锁:

[java] view plaincopy
 
  1. String hqlStr ="from TUser as user where user.name=‘Erica‘";  
  2. Query query = session.createQuery(hqlStr);  
  3. query.setLockMode("user",LockMode.UPGRADE); // 加锁  
  4. List userList = query.list();// 执行查询,获取数据  
[java] view plain copy
 
  1. String hqlStr ="from TUser as user where user.name=‘Erica‘";  
  2. Query query = session.createQuery(hqlStr);  
  3. query.setLockMode("user",LockMode.UPGRADE); // 加锁  
  4. List userList = query.list();// 执行查询,获取数据  

query.setLockMode对查询语句中,特定别名所对应的记录进行加锁(我们为TUser类指定了一个别名user),这里也就是对返回的所有 user 记录进行加锁。
观察运行期Hibernate生成的 SQL语句:

[sql] view plaincopy
 
  1. select tuser0_.id as id, tuser0_.name as name,tuser0_.group_id  
  2. as group_id, tuser0_.user_type as user_type, tuser0_.sex as sex  
  3. from t_user tuser0_ where (tuser0_.name=‘Erica‘ )for update  
[sql] view plain copy
 
  1. select tuser0_.id as id, tuser0_.name as name,tuser0_.group_id  
  2. as group_id, tuser0_.user_type as user_type, tuser0_.sex as sex  
  3. from t_user tuser0_ where (tuser0_.name=‘Erica‘ )for update  

这里Hibernate通过使用数据库的 for update子句实现了悲观锁机制。

hibernate的加锁模式有:
LockMode.NONE 
无锁机制。
LockMode.WRITE Hibernate Insert Update记录的时候会自动获取。
LockMode.READ Hibernate在读取记录的时候会自动获取。

以上这三种锁机制一般由 Hibernate内部使用,如Hibernate为了保证 Update过程中对象不会被外界修改,会在 save 方法实现中自动为目标对象加上WRITE锁。
LockMode.UPGRADE 利用数据库 for update 子句加锁。
LockMode. UPGRADE_NOWAIT  Oracle的特定实现,利用 Oracle for update nowait子句实现加锁。
上面这两种锁机制是我们在应用层较为常用的,加锁一般通过以下方法实现:
Criteria.setLockMode
Query.setLockMode
Session.lock

注意,只有在查询开始之前(也就是 Hiberate生成 SQL之前)设定加锁,才会真正通过数据库的锁机制进行加锁处理,否则,数据已经通过不包含 for update子句的 Select SQL加载进来,所谓数据库加锁也就无从谈起。

Hibernate使用悲观锁十分容易,但实际应用中悲观锁是很少被使用的,因为它大大限制了并发性,并且利用数据库底层来维护锁,这样大大降低了应用程序的效率。

下面我们来看一下hibernateAPI中提供的两个get方法:

GetClassclazzSerializable idLockMode lockMode

GetClassclazzSerializable idLockOptions lockOptions 

可以看到get方法第三个参数"lockMode""lockOptions",注意在Hibernate3.6以上的版本中"LockMode"已经不建议使用。方法的第三个参数就是用来设置悲观锁的,使用第三个参数之后,我们每次发送的SQL语句都会加上"for update"用于告诉数据库锁定相关数据。LockMode参数选择UPGRADE选项,就会开启悲观锁。

乐观锁(Optimistic Locking

      相对悲观锁而言,乐观锁机制采取了更加宽松的加锁机制。悲观锁大多数情况下依靠数据库的锁机制实现,以保证操作最大程度的独占性。但随之而来的就是数据库性能的大量开销,特别是对长事务而言,这样的开销往往无法承受。乐观锁机制在一定程度上解决了这个问题。乐观锁,大多是基于数据版本(Version)记录机制实现。何谓数据版本?即为数据增加一个版本标识,在基于数据库表的版本解决方案中,一般是通过为数据库表增加一个"version"字段来实现。
  乐观锁的工作原理:读取出数据时,将此版本号一同读出,之后更新时,对此版本号加一。此时,将提交数据的版本数据与数据库表对应记录的当前版本信息进行比对,如果提交的数据版本号大于数据库表当前版本号,则予以更新,否则认为是过期数据。

Hibernate为乐观锁提供了3中实现:

1.基于version

2.基于timestamp

3.为遗留项目添加添加乐观锁

配置基于version的乐观锁:

[html] view plaincopy
 
  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <!DOCTYPE hibernate-mapping PUBLIC"-//Hibernate/Hibernate Mapping DTD 3.0//EN""http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">  
  3.   
  4. <hibernate-mapping>  
  5.     <classnameclassname="com.bzu.hibernate.pojos.People"table="people">  
  6.         <idnameidname="id"type="string">  
  7.             <columnnamecolumnname="id"></column>  
  8.             <generatorclassgeneratorclass="uuid"></generator>  
  9.         </id>  
  10.         
  11.         <!--version标签用于指定表示版本号的字段信息-->  
  12.         <versionnameversionname="version"column="version"type="integer"></version>  
  13.   
  14.         <propertynamepropertyname="name"column="name"type="string"></property>  
  15.         
  16.     </class>  
  17. </hibernate-mapping>  

注:不要忘记在实体类添加属性version

下面我们就模拟多个session,基于version的来进行一下测试

[java] view plaincopy
 
  1. /* 
  2.          * 模拟多个session操作student数据表 
  3.          */  
  4.          
  5.         Sessionsession1=sessionFactory.openSession();  
  6.         Session session2=sessionFactory.openSession();  
  7.         Studentstu1=(Student)session1.createQuery("from Student s wheres.name='tom11'").uniqueResult();  
  8.         Studentstu2=(Student)session2.createQuery("from Student s wheres.name='tom11'").uniqueResult();  
  9.          
  10.         //这时候,两个版本号是相同的   
  11.        System.out.println("v1="+stu1.getVersion()+"--v2="+stu2.getVersion());  
  12.          
  13.         Transactiontx1=session1.beginTransaction();  
  14.        stu1.setName("session1");  
  15.         tx1.commit();  
  16.         //这时候,两个版本号是不同的,其中一个的版本号递增了   
  17.        System.out.println("v1="+stu1.getVersion()+"--v2="+stu2.getVersion());  
  18.          
  19.         Transactiontx2=session2.beginTransaction();  
  20.        stu2.setName("session2");  
  21.         tx2.commit();  

 

运行结果:

Hibernate: insert into studentVersion (ver, name,id) values (?, ?, ?)
Hibernate: select student0_.id as id0_, student0_.ver as ver0_, student0_.nameas name0_ from studentVersion student0_ where student0_.name='tom11'
Hibernate: select student0_.id as id0_, student0_.ver as ver0_, student0_.nameas name0_ from studentVersion student0_ where student0_.name='tom11'
v1=0--v2=0
Hibernate: update studentVersion set ver=?, name=? where id=? and ver=?
v1=1--v2=0
Hibernate: update studentVersion set ver=?, name=? where id=? and ver=?
Exception in thread "main" org.hibernate.StaleObjectStateException:Row was updated or deleted by another transaction (or unsaved-value mapping wasincorrect): [Version.Student#4028818316cd6b460116cd6b50830001]

 

     可以看到,第二个用户session2修改数据时候,记录的版本号已经被session1更新过了,所以抛出了红色的异常,我们可以在实际应用中处理这个异常,例如在处理中重新读取数据库中的数据,同时将目前的数据与数据库中的数据展示出来,让使用者有机会比较一下,或者设计程序自动读取新的数据

 

来源:http://blog.csdn.net/qq_27376871/article/details/51596749

 

 

 

 

分享到:
评论

相关推荐

    thinkPHP框架乐观锁和悲观锁实例分析

    在ThinkPHP框架中,常见的有乐观锁和悲观锁两种机制。本文将通过实例分析这两种锁在ThinkPHP框架中的应用,以及它们各自的优缺点。 首先,让我们从乐观锁开始了解。乐观锁机制假设多个事务在处理数据时很少发生冲突...

    面试必备之乐观锁与悲观锁.pdf

    ### 面试必备之乐观锁与悲观锁详解 #### 一、悲观锁与乐观锁的概念 悲观锁和乐观锁是计算机科学中用于处理并发控制的两种不同策略,它们主要应用于多线程环境下数据的一致性和完整性保护。两种锁的设计哲学反映了...

    乐观锁与悲观锁

    介绍数据库事务的定义和事务带来的问题,详细讲解乐观锁与悲观锁的区别

    Hibernate悲观锁和乐观锁的实现

    悲观锁和乐观锁是两种常见的锁定策略,它们各有特点,适用于不同的场景。 **悲观锁**(Pessimistic Lock)的名字来源于其悲观的态度,它认为数据随时可能被其他事务修改,因此在读取数据时就立即进行加锁,防止其他...

    Java 中的悲观锁和乐观锁的实现

    ### Java中的悲观锁与乐观锁实现详解 #### 一、悲观锁(Pessimistic Locking) 悲观锁是一种基于对数据安全性的保守态度而设计的锁机制。它假设数据在处理过程中很可能被外界修改,因此在整个数据处理过程中都会将...

    数据库事务、hibernate悲观锁和乐观锁

    在处理并发问题时,Hibernate提供了悲观锁和乐观锁两种机制。 悲观锁假设并发环境中的冲突是常态,因此在读取数据时就立即锁定,直到事务结束才释放。在Hibernate中,可以通过设置`@LockModeType.PESSIMISTIC_READ`...

    Hibernate的乐观锁与悲观锁

    ### Hibernate的乐观锁与悲观锁 #### 一、引言 在并发环境下,尤其是在金融、电商等业务场景中,确保数据的一致性和完整性至关重要。**Hibernate**作为一种流行的Java持久层框架,提供了多种机制来处理并发控制...

    Hibernate乐观锁和悲观锁分析

    【Hibernate乐观锁与悲观锁详解】 在开发过程中,尤其是在并发环境下,确保数据的一致性和完整性至关重要。Hibernate,作为Java领域广泛使用的ORM框架,提供了一种处理并发数据访问冲突的手段,那就是锁机制。主要...

    SpringBoot整合MyBatis实现乐观锁和悲观锁的示例

    SpringBoot整合MyBatis实现乐观锁和悲观锁的示例 在本文中,我们将学习如何使用SpringBoot和MyBatis来实现乐观锁和悲观锁。我们将通过示例代码来介绍这两种锁的实现方式,帮助读者更好地理解和使用它们。 一、悲观...

    Hibernate悲观锁与乐观锁案例

    在Java的持久化框架Hibernate中,悲观锁和乐观锁是两种重要的并发控制策略,它们用于管理数据库中的数据在多线程环境下的访问安全。本文将深入探讨这两种锁机制的原理、应用场景及其区别。 首先,我们来理解悲观锁...

    并发控制—悲观锁和乐观锁

    并发控制—悲观锁和乐观锁

    hibernate乐观锁和悲观锁学习

    本文主要讨论的是Hibernate框架中两种锁机制的使用:乐观锁和悲观锁。 首先,让我们深入理解悲观锁(Pessimistic Locking)。悲观锁正如其名字所示,假设并发环境中数据会被频繁修改,所以在整个数据处理过程中,它...

    Hibernate锁机制_悲观锁和乐观锁

    Hibernate 锁机制_悲观锁和乐观锁 Hibernate 锁机制是指在数据库访问中,为了维护数据的一致性和正确性,所采取的一些机制来防止数据的并发访问和修改。 Hibernate 中有两种锁机制:悲观锁和乐观锁。 一、悲观锁...

    Hibernate悲观锁与乐观锁

    《Hibernate 悲观锁与乐观锁详解》 在多用户并发访问的环境中,数据库管理系统必须具备有效的数据访问控制机制,以确保数据的一致性和完整性。Hibernate,作为一款流行的Java持久化框架,提供了两种主要的锁定策略...

    悲观锁和乐观锁的技术对比和应用.docx

    悲观锁和乐观锁是两种常见的并发控制策略,用于在多任务环境下确保数据一致性。它们各自有不同的工作原理和适用场景。 悲观锁(Pessimistic Lock)正如其名,它假设并发环境中会发生频繁的冲突,因此在读取数据时就...

    面试必备之乐观锁与悲观锁

    【标题】:“面试必备之乐观锁与悲观锁” 【描述】:“面试必备之乐观锁与悲观锁.pdf”涉及的是并发控制中的两种重要锁机制——悲观锁和乐观锁,它们是多线程环境下确保数据一致性的重要手段。 【标签】:“求职...

    悲观锁和乐观锁.md

    所谓悲观锁,总是假设最坏的情况,每次去拿数据的时候都会认为别人会修改数据,造成幻读,不可重复读,脏读等情况发生。所谓乐观锁,重视假设最好的情况,每次去拿数据都认为别人不会修改,所以不会上锁,但是会在...

    Java并发篇乐观锁,悲观锁,自旋锁

    本文主要讨论了四种锁类型:乐观锁、悲观锁、自旋锁以及Java中的synchronized同步锁,并深入解析了synchronized锁的内部机制,包括其核心组件、实现方式以及锁的状态。 1. **乐观锁**:乐观锁假设在多线程环境下,...

Global site tag (gtag.js) - Google Analytics