hibernate 对悲观锁及乐观锁的支持

在实际应用中常常会出现这种情况,两个人同时读取同一条数据,然后其中一个人对这条数据进行了修改,之后另外一个人也对这条数据进行修改,后来修改数据的人会覆盖先修改那个人的数据,这种情况叫做数据更新丢失。

hibernate提供两种方式来解决这种问题:悲观锁和乐观锁的方式。

悲观锁:

悲观锁的实现,通常依赖于数据库机制,在整个过程中将数据锁定,其它任何用户都不能读取或修改,即以独占的方式。例如微软的VSS。

乐观锁:

乐观锁大多数基于数据版本记录机制(version)实现,一般是在数据库表中加入一个version字段,读取数据时将版本号一同读出,之后更新数据时版本号加一,如果提交数据时版本号小于或等于数据表中的版本号,则认为数据是过期的,否则给予更新。即以版本的方式。例如CVS.

hibernate悲观锁例子:

1.实体类:

package com.yx.zzg.pojo;

public class Inventory {

private int id;

private String itemName;

private int quantity;

public int getId() {
return id;
}

public void setId(int id) {
this.id = id;
}

public String getItemName() {
return itemName;
}

public void setItemName(String itemName) {
this.itemName = itemName;
}

public int getQuantity() {
return quantity;
}

public void setQuantity(int quantity) {
this.quantity = quantity;
}

}


2.映射文件

<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping package="com.yx.zzg.pojo">
<class name="Inventory" table="t_inventory">
<id name="id">
<!-- 指定主键生成策略 -->
<generator class="native" />
</id>
<property name="itemName" />
<property name="quantity" />
</class>

</hibernate-mapping>


3.hibernate配置文件

<!DOCTYPE hibernate-configuration PUBLIC
"-//Hibernate/Hibernate Configuration DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">

<hibernate-configuration>
<session-factory>
<property name="connection.driver_class">com.mysql.jdbc.Driver</property>
<property name="connection.url">jdbc:mysql://localhost/hibernate_pessimistic_lock</property>
<property name="connection.username">root</property>
<property name="connection.password">123456</property>
<property name="hibernate.dialect">org.hibernate.dialect.MySQLDialect</property>
<!-- 是否显示执行的sql语句 -->
<property name="show_sql">true</property>
<!-- 显示执行的sql语句是否格式化 -->
<property name="format_sql">true</property>
<!-- 是否自动创建表,参数详见HIBERNATE_HOME/etc/hibernate.properties文件 -->
<!--
<property name="hibernate.hbm2ddl.auto">create</property>
-->
<mapping resource="com/yx/zzg/pojo/Inventory.hbm.xml" />
</session-factory>
</hibernate-configuration>


4.测试类:

package com.yx.zzg.test;

import java.util.Date;
import java.util.Iterator;
import java.util.Set;

import org.hibernate.HibernateException;
import org.hibernate.LockMode;
import org.hibernate.Session;
import org.hibernate.Transaction;

import com.yx.zzg.pojo.Inventory;
import com.yx.zzg.util.HibernateUtil;

import junit.framework.TestCase;

/**
* 测试方法:在testLoad1方法中打一个断点,运行到更新前停止,开始运行testLoad2方法,会发现testLoad2方法中的不会打印数据,
* 当testLoad1一方法运行完之后testLoad2方法才会继续运行
*
* @author toshiba
*
*/
public class TestUser extends TestCase {

public void testLoad1() {
Session session = null;
Transaction tx = null;
try {
session = HibernateUtil.getSession();
tx = session.beginTransaction();
// 发出sql语句,lazy失效
//LockMode.UPGRADE指示独占的方式(悲观锁)
Inventory inventory = (Inventory) session.load(Inventory.class, 1,
LockMode.UPGRADE);
// 不发sql
System.out.println("ItemName=" + inventory.getItemName());
System.out.println("Quantity=" + inventory.getQuantity());

inventory.setQuantity(inventory.getQuantity() - 200);

tx.commit();
} catch (HibernateException e) {
throw e;
} finally {
if (session != null) {
session.close();
}
}
}

public void testLoad2() {
Session session = null;
Transaction tx = null;
try {
session = HibernateUtil.getSession();
tx = session.beginTransaction();

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

System.out.println("ItemName=" + inventory.getItemName());
System.out.println("Quantity=" + inventory.getQuantity());

inventory.setQuantity(inventory.getQuantity() - 200);

tx.commit();
} catch (HibernateException e) {
throw e;
} finally {
if (session != null) {
session.close();
}
}
}

}



hibernate乐观锁例子:

1.实体类:

package com.yx.zzg.pojo;

public class Inventory {

private int id;

private String itemName;

private int quantity;

private int version;

public int getVersion() {
return version;
}

public void setVersion(int version) {
this.version = version;
}

public int getId() {
return id;
}

public void setId(int id) {
this.id = id;
}

public String getItemName() {
return itemName;
}

public void setItemName(String itemName) {
this.itemName = itemName;
}

public int getQuantity() {
return quantity;
}

public void setQuantity(int quantity) {
this.quantity = quantity;
}

}


2.映射文件

<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping package="com.yx.zzg.pojo">
<!-- optimistic-lock="version"指示该实体的加载是需要乐观锁的 ,这个乐观锁的支持是根据版本号-->
<class name="Inventory" table="t_inventory" optimistic-lock="version">
<id name="id">
<!-- 指定主键生成策略 -->
<generator class="native" />
</id>

<!-- 指示版本字段,该字段会有hibernate来维护 -->
<!-- version标签必须放在id标签下面 -->
<version name="version" />
<property name="itemName" />
<property name="quantity" />
</class>

</hibernate-mapping>


3.测试类:

package com.yx.zzg.test;

import org.hibernate.HibernateException;
import org.hibernate.Session;
import org.hibernate.Transaction;

import com.yx.zzg.pojo.Inventory;
import com.yx.zzg.util.HibernateUtil;

import junit.framework.TestCase;

/**
* 测试方法:在testLoad1方法中打一个断点,运行到更新前停止,开始运行testLoad2方法,这时testLoad2方法能正确执行,
* 能够打印出结果也能正常更新数据库中的数据
* ,并且将数据库中的version字段更新+1.此时接着运行testLoad1方法,则会抛出异常,因为这时testLoad1方法更新的是已过时的数据。
*
* @author toshiba
*
*/
public class TestUser extends TestCase {

public void testLoad1() {
Session session = null;
Transaction tx = null;
try {
session = HibernateUtil.getSession();
tx = session.beginTransaction();
// 不会发出sql。
Inventory inventory = (Inventory) session.load(Inventory.class, 1);
// 发出sql,正常打印
System.out.println("ItemName=" + inventory.getItemName());
System.out.println("Quantity=" + inventory.getQuantity());

inventory.setQuantity(inventory.getQuantity() - 200);

tx.commit();
} catch (HibernateException e) {
throw e;
} finally {
if (session != null) {
session.close();
}
}
}

public void testLoad2() {
Session session = null;
Transaction tx = null;
try {
session = HibernateUtil.getSession();
tx = session.beginTransaction();

Inventory inventory = (Inventory) session.load(Inventory.class, 1);
// 能够正常打印
System.out.println("ItemName=" + inventory.getItemName());
System.out.println("Quantity=" + inventory.getQuantity());
inventory.setQuantity(inventory.getQuantity() - 200);
// 发出一条根据ID和版本号来更新数据的sql语句
tx.commit();
} catch (HibernateException e) {
throw e;
} finally {
if (session != null) {
session.close();
}
}
}

}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值