点击蓝色“程序猿DD”关注我哟
加个“星标”,不忘签到哦
来源:锅外的大佬
关注我,回复口令获取可获取独家整理的学习资料:
- 001:领取《Spring Boot基础教程》
- 002:领取《Spring Cloud基础教程》
- 003:领取《Java开发规范1.5》(最新版)
1. 概述
在本快速教程中,我们将讨论在Spring Data JPA中为自定义查询方法和预定义存储库的 CRUD
方法启用事务锁, 我们还将查看不同的锁类型并设置事务锁超时。
2. 锁类型
JPA定义了两种主要的锁类型,即悲观锁和乐观锁。
2.1 悲观锁
当我们在事务中使用悲观锁并访问实体时,它将立即锁定。通过提交或回滚事务来释放锁。
2.2 乐观锁
在乐观锁中,事务不会立即锁定实体。相反,事务通常会保存实体的状态,并为其分配版本号。
当我们尝试在不同的事务中更新实体的状态时,事务会在更新期间将保存的版本号与现有的版本号进行比较。
此时,如果版本号不同,则表示无法修改实体。如果存在活动事务,那么该事务将被回滚,并且底层JPA实现将抛出OptimisticLockException。
除版本号方法外,我们还可以使用其他方法,如时间戳,哈希值计算或序列化校验和,具体取决于哪种方法最适合我们当前的开发环境。
3. 在查询方法上启用事务锁
要获取实体的锁定,我们可以通过使用 @Lock
来注解目标查询方法, 并传递所需的锁定模式类型。
锁定模式类型(Lock mode types)是锁定实体时要指定的枚举值。然后,将指定的锁定模式传递到数据库,以在实体对象上应用相应的锁定。
要在Spring Data JPA存储库的自定义查询方法上指定锁定,我们可以使用 @Lock
注解该方法并指定所需的锁定模式类型:
@Lock(LockModeType.OPTIMISTIC_FORCE_INCREMENT)
@Query("SELECT c FROM Customer c WHERE c.orgId = ?1")
public List<Customer> fetchCustomersByOrgId(Long orgId);
要强制锁定预定义的存储库方法(如findAll或findById(id)),我们必须在存储库中声明方法并使用 @Lock
注解该方法:
@Lock(LockModeType.PESSIMISTIC_READ)
public Optional<Customer> findById(Long customerId);
当显式启用锁并且没有活动事务时,底层JPA实现将抛出TransactionRequiredException。
如果无法授予锁并且锁冲突不会导致事务回滚,则JPA会抛出LockTimeoutException。但它不标记回滚的活动事务。
4. 设置事务锁定超时
使用悲观锁时,数据库将尝试立即锁定实体。当无法立即获取锁定时,底层JPA实现会抛出LockTimeoutException。为避免此类异常,我们可以指定锁超时时间。
在Spring Data JPA中,可以使用@QueryHints指定锁定超时时间:
@Lock(LockModeType.PESSIMISTIC_READ)
@QueryHints({@QueryHint(name = "javax.persistence.lock.timeout", value = "3000")})
public Optional<Customer> findById(Long customerId);
有关在不同范围设置锁定超时提示的更多详细信息,请参见ObjectDB文章。
5. 结论
在本教程中,我们学习了不同类型的事务锁定模式,以及如何在Spring Data JPA中启用事务锁,并简单介绍了如何设置锁定超时。
在正确的位置应用正确的事务锁,可以帮助维护高并发情况下应用程序中数据完整性。
当交易需要严格遵守ACID规则时,我们应该使用悲观锁。当我们需要允许多个并发读取以及在应用程序上下文中可接受的最终一致性时,应该应用乐观锁。
当然,可以在Github上找到悲观锁和乐观锁的示例代码。
新群招募
如果你喜欢自己装机、刷系统、玩黑群,或对各类科技产品有浓厚的兴趣!那么赶紧加入,一起聊聊喜欢的东西,在大促的时候种种草。亦或是聊聊你想买的东西,让大伙给你拔拔草。
添加微信:zyc_enjoy,根据指引,加入讨论群
推荐阅读:
活动介绍:自律到极致-人生才精致:第9期
活动奖励:
一等奖:天猫精灵 * 1
二等奖:我的星球会员 * 5
扫描下面二维码签到参与
关注我,加个星标,不忘签到哦~
推荐关注
推荐一个专注于国外大佬文章的翻译与发布的公众号。通过这个公众号,可以帮助我们紧跟技术的国际化发展趋势。
点一点“阅读原文”小惊喜在等你