hibernate学习之三(悲观锁与乐观锁)

简介: hibernate学习之三(悲观锁与乐观锁)

悲观锁与乐观锁


悲观锁

悲观锁(pessimistic lock)是指在每次操作数据时,总是悲观地认为会有其他事务操作同一数据,因此,在整个数据处理过程中,会把数据处于锁定状态。


悲观锁具有排他性,一般由数据库实现。在锁定时间内,其他事务不能对数据进行存取等操作,这可能导致长时间的等待问题。


通常在应用中会设定如下两种锁模式。


1)LockMode.UPGRADE


该模式不管缓存中是否存在对象,总是通过 select 语句到数据库中加载该对象,如果映射文件中设置了版本元素,则执行版本检查,比较缓存中的对象是否与数据库中对象的版本一致,如果数据库系统支持悲观锁(如 MySQL),则执行 elect…for update 语句,如果不支持(如 Sybase),则执行普通 select 语句。


2)LockMode.UPGRADE_NOWAIT


该模式与 LockMode.UPGRADE 具有同样的功能,是 Oracle 数据库特有的锁模式,会执行 select…for update nowait 语句。


nowait 表示如果执行 select 语句的事务不成立则获得悲观锁,它不会等待其他事务释放锁,而是立刻抛出锁定异常。


下面通过丢失更新的案例演示悲观锁的使用。


案例主要代码:


@Getter
@Setter
public class Inventory {
private int itemNo;
private String itemName;
private int quantity;
}


<hibernate-mapping>
   <class name="com.bjsxt.hibernate.Inventory" table="t_inventory">
      <id name="itemNo">
         <generator class="native"/>
      </id>
      <property name="itemName"/>
      <property name="quantity"/>
   </class>
</hibernate-mapping>


使用:


Inventory inv = (Inventory)session.load(Inventory.class, 1, LockMode.UPGRADE);


悲观锁不支持懒加载,也就是说一加锁懒加载就失效了


乐观锁


相对于悲观锁而言,乐观锁(optimistic lock)通常认为多个事务同时操作同一数据的情况很少发生,因此乐观锁不进行数据库层次上的锁定,而是基于数据版本(Version)标识实现应用程序级别上的锁定机制,这既能保证多个事务的并发操作,又能有效防止第二类丢失更新的发生。


数据版本标识是通过为数据表增加一个 version 字段实现的。增加 version 字段后,程序在读取数据时,会将版本号一同读出,之后在更新此数据时,会将此版本号加一。


在提交数据时,将现有的版本号与数据表对应记录的版本号进行对比,如果提交数据的版本号大于数据表中的版本号,则允许更新数据,否则禁止更新数据。


主要代码:


 @Setter
 @Getter
 public class Inventory {
   private int itemNo;
   private String itemName;
   private int quantity;
   private int version1;
}


<hibernate-mapping>
   <class name="com.bjsxt.hibernate.Inventory" table="t_inventory" optimistic-lock="version">
      <id name="itemNo">
         <generator class="native"/>
      </id>
      <version name="version1"/>
      <property name="itemName"/>
      <property name="quantity"/>
   </class>
</hibernate-mapping>


Hibernate 的乐观锁是以 id 和 version 决定更新对象的。

相关文章
|
3月前
|
数据库 开发者 Java
数据战争:Hibernate的乐观与悲观锁之争,谁将主宰并发控制的王座?
【8月更文挑战第31天】在软件开发中,数据一致性至关重要,尤其是在多用户并发访问环境下。Hibernate 作为 Java 社区常用的 ORM 框架,提供了乐观锁和悲观锁机制来处理并发问题。乐观锁假设数据不易冲突,通过版本号字段 (`@Version`) 实现;悲观锁则假定数据易冲突,在读取时即加锁。选择哪种锁取决于具体场景:乐观锁适合读多写少的情况,减少锁开销;悲观锁适合写操作频繁的场景,避免数据冲突。正确应用这些机制可提升应用程序的健壮性和效率。
36 0
|
5月前
|
Java 数据库连接 数据库
JPA和Hibernate的乐观锁与悲观锁
木头左讲解JPA和Hibernate中的乐观锁与悲观锁。乐观锁在并发更新少、冲突处理成本高、数据一致性要求不严的场景下适用,利用`@Version`注解防止冲突。而悲观锁适合并发更新频繁、处理冲突成本低、需高度数据一致性的场景,通过`@Lock`注解实现锁机制。选择合适的锁策略对提升数据库性能和保证数据一致性至关重要。
JPA和Hibernate的乐观锁与悲观锁
|
5月前
|
Java 数据库连接 API
解锁你的数据库:JPA和Hibernate的乐观锁与悲观锁
本文由木头左介绍JPA和Hibernate中的乐观锁与悲观锁。乐观锁假设无冲突,通过`@Version`注解防止并发更新,适用于更新不频繁、处理冲突成本高、数据一致性要求不高的场景。悲观锁假设有冲突,利用`@Lock`注解实现加锁,适用于并发更新频繁、处理冲突成本低、数据一致性要求高的情况。选择哪种锁取决于具体需求。
解锁你的数据库:JPA和Hibernate的乐观锁与悲观锁
|
Java 数据库连接 数据库
锁机制有什么用?简述Hibernate的悲观锁和乐观锁机制
有些业务逻辑在执行过程中要求对数据进行排他性的访问,于是需要通过一些机制保证在此过程中数据被锁住不会被外界修改,这就是所谓的锁机制。 Hibernate支持悲观锁和乐观锁两种锁机制。
1607 0
|
Java 数据库连接 数据库
Hibernate学习之Hibernate注解总结
Hibernate学习之Hibernate注解总结http://www.bieryun.com/3269.html 一、类级别的注解 @Entity name:表的名字(可选)一般表名和类名相同 必须指定主键属性@Id @Table name:映射表的名称(可选) catalog:目录(可选)默认为空 schema:模式(可选)默认为空 与@Entity注解配合使用,只能表示在实体类class定义处,表示实体类对应数据库表的信息 @Embeddable 表示一个非Entity类,不是一个实体类,可以嵌入到实体类中作为一个属性存在。
1659 0
|
Java 数据库连接 数据库
Hibernate学习之Hibernate注解总结
Hibernate学习之Hibernate注解总结 一、类级别的注解 @Entity name:表的名字(可选)一般表名和类名相同 必须指定主键属性@Id @Table name:映射表的名称(可选) catalog:目录(可选)默认为空 schema:模式(可选)默认为空 与@Entity注解配合使用,只能表示在实体类class定义处,表示实体类对应数据库表的信息 @Embeddable 表示一个非Entity类,不是一个实体类,可以嵌入到实体类中作为一个属性存在。
1355 0
|
Java 数据库连接 数据库