当前位置: 首页 > news >正文

Java--乐观锁

乐观锁是一种并发控制机制,用于处理多个事务或线程对同一数据进行并发修改的问题。它假设多个事务或线程在操作数据时不会互相干扰,因此不在数据上加锁,而是在提交数据时检查数据是否被其他事务修改过。如果数据在提交前已经被其他事务修改,则当前事务需要重新读取数据并尝试再次提交。乐观锁的核心思想是“乐观地”认为数据冲突的概率很低,因此主要在提交阶段进行冲突检测。

乐观锁的实现方式

乐观锁的常见实现方式是使用版本号或时间戳:

  1. 版本号(Version Number)

    • 在数据表中增加一个版本号字段,每当数据被修改时,版本号加1。
    • 事务在读取数据时,会同时读取版本号。
    • 在更新数据时,事务会检查当前数据的版本号是否与读取时的版本号一致。如果一致,则进行更新并将版本号加1;如果不一致,则说明数据已经被其他事务修改,当前事务需要重新读取数据再进行处理。
  2. 时间戳(Timestamp)

    • 在数据表中增加一个时间戳字段,记录数据的最后修改时间。
    • 事务在读取数据时,会同时读取时间戳。
    • 在更新数据时,事务会检查当前数据的时间戳是否与读取时的时间戳一致。如果一致,则进行更新并更新时间戳;如果不一致,则说明数据已经被其他事务修改,当前事务需要重新读取数据再进行处理。

示例代码

以下是一个使用版本号实现乐观锁的示例:

数据表设计
CREATE TABLE user (id BIGINT PRIMARY KEY,name VARCHAR(50),balance DECIMAL(10, 2),version INT
);
实体类
public class User {private Long id;private String name;private BigDecimal balance;private Integer version;// Getters and Setters
}
Mapper 接口
public interface UserMapper extends BaseMapper<User> {@Update("UPDATE user SET balance = #{balance}, version = version + 1 WHERE id = #{id} AND version = #{version}")int updateUser(User user);
}
服务实现类
@Service
public class UserServiceImpl extends ServiceImpl<UserMapper, User> implements UserService {@Transactional@Overridepublic void deductBalance(Long id, BigDecimal money) {// 1.查询用户User user = getById(id);if (user == null) {throw new RuntimeException("用户不存在");}// 2.校验用户状态和余额if (user.getBalance().compareTo(money) < 0) {throw new RuntimeException("用户余额不足");}// 3.扣减余额user.setBalance(user.getBalance().subtract(money));// 4.尝试更新用户信息int updateCount = baseMapper.updateUser(user);if (updateCount == 0) {// 如果更新失败,说明版本号不一致,需要重新读取数据并重试throw new RuntimeException("更新失败,请重试");}}
}

适用场景和优缺点

适用场景

  • 适用于读多写少的应用场景,例如电商系统中的商品库存管理。
  • 适用于不希望在数据上加锁,减少锁开销,提高并发性能的场景。

优点

  • 无需加锁,减少了锁开销,提高了系统并发性能。
  • 避免了死锁的发生。

缺点

  • 在写操作频繁的场景下,重试的代价较高,可能影响性能。
  • 实现复杂度较高,需要在应用程序中额外处理冲突重试逻辑。

总结

乐观锁是一种有效的并发控制机制,通过版本号或时间戳实现冲突检测,适用于读多写少的场景,能提高系统的并发性能。在实际应用中,需要根据具体业务场景选择合适的并发控制策略。

http://www.lryc.cn/news/383760.html

相关文章:

  • 静默升级oracle 11g (从11.2.0.1升级到11.2.0.4)
  • 什么是模型训练,如何选择合适的Batch大小
  • 【线上绘图网站分享】
  • Snipaste截图工具如何控制框线箭头的粗细程度
  • GISSERVER 管理器 1.0(私有化地图离线部署)
  • Eureka服务治理深度解析:服务下线与剔除机制揭秘
  • 苹果笔记本双系统怎么安装
  • 探索网络爬虫技术:原理、实践与挑战
  • GitHub国内使用方法
  • Java调用第三方HTTP接口的常用方式
  • DOPE-PEG2000-FITC荧光特性
  • 华为Pura70支持5G功能吗?看完你就清楚了
  • android 4大组件用法
  • qt pro工程文件通用宏定义
  • 这次让我们隆重的介绍一下
  • 大语言模型系列-Transformer
  • Node.js 语言特定指南
  • 科普:什么是 BC-404 ?全方位解读最新通缩型 NFT 标准
  • 软件测试学习笔记丨JUnit5执行顺序
  • 解决GPU 显存未能完全释放
  • 3D资产爆发,轻量化需求再度冲高,见证下一代3D崛起!
  • AI绘画的10种变现方法,逼你躺平挣钱
  • Pura 70 系列超高速风驰闪拍,捕捉美好,告别抓拍模糊
  • AI作画Prompt不会写?Amazon Bedrock Claude3.5来帮忙
  • SSL证书类型解析:DV、OV、EV证书的区别与适用场景
  • WPF 2024 金九银十 最新 高级 架构 面试题 C#
  • Linux上使用 git 命令行
  • vue 中computed和watch的区别
  • 富豪王思聪的“爱情喜剧”从万达排片到网红聊天
  • qt qml-http之XMLHttpRequest介绍详解使用