Hibernate乐观锁与悲观锁详解

需积分: 17 5 下载量 79 浏览量 更新于2024-09-19 收藏 36KB DOC 举报
"这篇文档主要介绍了Hibernate框架中的两种锁机制——乐观锁和悲观锁,以及它们在实际业务场景中的应用。文档首先展示了如何在Hibernate映射文件中设置乐观锁属性,然后详细解释了乐观锁的不同实现方式,推荐使用版本机制。接着,文档描述了悲观锁的概念,它是通过数据库的锁机制确保数据的排他访问,并给出了一个Hibernate实现悲观锁的示例。" 在Java持久化框架Hibernate中,锁机制对于确保数据一致性至关重要。乐观锁和悲观锁是两种常见的策略,它们各有优缺点,适用于不同的业务需求。 1. **乐观锁**: - 乐观锁假设并发环境中,数据冲突的概率较低,因此在读取数据时不进行锁定,只有在更新数据时才会检查在此期间数据是否被其他事务修改过。如果发现数据已被修改,则更新失败。 - Hibernate提供了多种乐观锁实现方式,包括`none`、`version`、`dirty`和`all`。其中,`version`是最推荐的,通过增加一个版本字段,每次数据更新时版本号递增,如果读取时版本号与更新时不同,就认为数据已被其他事务修改,从而防止脏读。 - 在给定的描述中,`<class>`标签中的`optimistic-lock="version"`就是开启基于版本号的乐观锁机制,它是最安全且高效的方式,尤其适用于读多写少的场景。 2. **悲观锁**: - 与乐观锁相反,悲观锁假设并发环境中数据冲突可能性较高,所以在读取数据时立即进行锁定,直到事务结束才释放锁,确保数据在处理过程中不会被其他事务修改。 - 数据库层面的悲观锁通常是通过`SELECT ... FOR UPDATE`语句实现,这会在查询时锁定选定的记录,直到当前事务结束。 - Hibernate中,可以通过`Query.setLockMode()`方法来设置悲观锁,例如`LockMode.UPGRADE`表示获取升级锁,相当于数据库的行级锁,这样可以确保在查询后更新数据时不会与其他事务发生冲突。 - 示例代码演示了如何在Hibernate查询中使用悲观锁,通过设置`setLockMode`对查询结果进行加锁,从而保证数据的排他访问。 在实际应用中,选择哪种锁机制取决于业务场景。乐观锁适合读多写少且数据冲突概率低的情况,减少了因加锁导致的阻塞和性能损失。而悲观锁更适合写多或者对数据一致性要求极高的场景,它能保证在处理数据时不会出现其他事务的干扰。开发者需要根据业务需求权衡锁的选择,以达到最佳的并发控制效果。

Batch update returned unexpected row count from update [0]; actual row count: 0; expected: 1 org.hibernate.StaleStateException: Batch update returned unexpected row count from update [0]; actual row count: 0; expected: 1 at org.hibernate.jdbc.Expectations$BasicExpectation.checkBatched(Expectations.java:61) at org.hibernate.jdbc.Expectations$BasicExpectation.verifyOutcome(Expectations.java:46) at org.hibernate.jdbc.BatchingBatcher.checkRowCounts(BatchingBatcher.java:68) at org.hibernate.jdbc.BatchingBatcher.doExecuteBatch(BatchingBatcher.java:48) at org.hibernate.jdbc.AbstractBatcher.executeBatch(AbstractBatcher.java:246) at org.hibernate.engine.ActionQueue.executeActions(ActionQueue.java:235) at org.hibernate.engine.ActionQueue.executeActions(ActionQueue.java:140) at org.hibernate.event.def.AbstractFlushingEventListener.performExecutions(AbstractFlushingEventListener.java:298) at org.hibernate.event.def.DefaultAutoFlushEventListener.onAutoFlush(DefaultAutoFlushEventListener.java:41) at org.hibernate.impl.SessionImpl.autoFlushIfRequired(SessionImpl.java:969) at org.hibernate.impl.SessionImpl.listCustomQuery(SessionImpl.java:1690) at org.hibernate.impl.AbstractSessionImpl.list(AbstractSessionImpl.java:142) at org.hibernate.impl.SQLQueryImpl.list(SQLQueryImpl.java:152) at com.sinosoft.sysframework.persistence.HibernateSQLQuery.doInHibernate(HibernateSQLQuery.java:118) at org.springframework.orm.hibernate3.HibernateTemplate.execute(HibernateTemplate.java:366) at org.springframework.orm.hibernate3.HibernateTemplate.executeFind(HibernateTemplate.java:338) at com.sinosoft.sysframework.persistence.BaseDaoHibernateImpl.queryList(BaseDaoHibernateImpl.java:224) at com.sinosoft.application.reins.persistence.hibernate.base.GrRePlanItemDaoHibernateImplBase.find(GrRePlanItemDaoHibernateImplBase.java:127) at com.sinosoft.application.reins.service.spring.GrTrialServiceSpringImpl.endorTrialDataSurrender(GrTrialServiceSpringImpl.java:9536) at com.sinosoft.application.reins.service.

193 浏览量