OptimisticLock

        想象你和你的朋友去了一家很受欢迎的餐厅。你们想要点一份特别的菜品——这家餐厅的招牌菜,但因为这道菜非常受欢迎,所以它的状态可能会随时变化(比如售罄或重新上架)。

传统方式(悲观锁)

        通常情况下,服务员会先查看厨房是否还有这道菜(检查库存)。如果有,服务员就会立即告诉厨房停止接受更多对该菜品的订单,并为你们下单。这样做是为了确保在你们下单期间,没有人能抢走最后一份这道菜。这就是悲观锁的思想——提前锁定资源以防止其他请求干扰。

乐观方式

        但是,这家餐厅采用了更灵活的方式处理订单。他们允许顾客先选择菜品并准备好付款,而不立即通知厨房。当你们准备好了要付款时,服务员再去确认一次厨房里是否有这道菜。如果在这段时间内菜品的状态没有改变(比如仍然有库存),那么你们就可以顺利下单;但如果菜品已经售罄了,服务员会告诉你们这个消息,并建议你们选择其他菜品。

这种方法就是乐观锁的核心思想:

  • 假设冲突很少发生
    • 大多数时候,你们应该能够成功点到这道菜,因为在你们选择菜品和实际下单之间的时间间隔很短,而且餐厅有足够的库存。
  • 最后时刻验证
    • 只有在你们真正决定下单的时候,才会去检查菜品的状态是否与之前看到的一致。
  • 冲突处理
    • 如果发现状态不一致(例如菜品已售罄),则当前操作失败,需要用户做出新的选择。

Java中的乐观锁应用

在Java编程中,乐观锁通常通过版本号、时间戳或其他标记来实现。下面是一个简化的示例,展示如何使用版本号来进行乐观锁控制:

public class MenuItem {private String name;private int version; // 用于乐观锁的版本号public boolean order() {// 获取最新的菜单项信息MenuItem latest = getLatestMenuItemFromDatabase();// 假设我们从界面上获取到了用户选择时的版本号int userSelectedVersion = getUserSelectedVersion();// 如果版本号匹配,说明自用户选择以来没有其他人修改过这条记录if (latest.getVersion() == userSelectedVersion) {// 更新数据库中的记录并增加版本号updateMenuItemInDatabase(latest, latest.getVersion() + 1);return true; // 订单成功} else {// 版本号不匹配,说明数据已被修改return false; // 订单失败,提示用户重试}}
}

        在这个例子中,order() 方法尝试根据用户选择时的版本号来完成订单。它首先获取最新的菜单项信息,然后检查版本号是否匹配。如果匹配,则更新数据库并增加版本号,表示订单成功;如果不匹配,则返回 false,意味着订单由于数据被其他用户修改而失败,需要用户重新选择或者刷新页面再试一次。

        通过这种方式,乐观锁允许更高的并发性,因为它不会阻止其他用户同时访问相同的数据,只是在最后一步确保数据一致性。如果确实发生了冲突,系统会要求用户重试他们的操作。这种机制非常适合读多写少的应用场景,如在线商店的商品详情页、论坛帖子等。

我可以理解 这就是普通的业务判断是吗?

        乐观锁不仅仅是简单的业务判断,它是一种并发控制策略,用于在多用户环境下确保数据的一致性和完整性。虽然从表面上看,乐观锁的实现可能看起来像是普通的业务逻辑判断(比如检查版本号是否匹配),但它背后的理念和作用更为深远。

乐观锁的核心特点

        假设冲突很少发生:乐观锁假定多个事务可以同时进行而不会相互干扰,因此不需要提前锁定资源。只有在提交时才会检查是否有冲突发生。

        最后时刻验证:当一个事务准备提交时,它会检查自开始以来该资源的状态是否发生了变化。如果状态没有改变,则允许提交;如果有变化,则认为发生了冲突,事务需要回滚或重试。

        高并发下的高效性:由于不需要长时间持有锁,乐观锁允许更高的并发度,特别适用于读操作远多于写操作的应用场景

        轻量级且非阻塞:与悲观锁相比,乐观锁不涉及锁机制,因此不会造成线程等待或阻塞的情况,从而提高了系统的响应速度。

        冲突处理:当检测到冲突时,通常会提示用户重新尝试操作或者系统自动重试。

为什么这不是普通的业务判断?

        并发控制:乐观锁是专门为了解决并发环境下的数据一致性问题而设计的。它的主要目的是避免多个事务同时修改同一数据而导致的数据不一致或丢失更新的问题。

        性能优化:通过减少锁的使用,乐观锁可以在高并发的情况下提供更好的性能。相比之下,普通的业务判断通常不会考虑这些并发问题,也不会试图优化这种场景下的性能。

        数据完整性保障:乐观锁确保了即使在高并发的情况下,也能维持数据的完整性和一致性,而这并不是普通业务判断所能提供的保证。

结合上面餐馆点餐例子

        在餐馆点餐的例子中,乐观锁不仅仅是在最后检查菜品是否还有库存(这确实类似于普通的业务判断)。更重要的是,它允许多位顾客几乎同时选择同一道菜而不必担心其中一方会被无故阻塞。只有到了真正下单的时候,才会去确认菜品的状态。如果在这期间菜品状态改变了(例如被其他人订购完了),那么当前的订单就会失败,并提示顾客重新选择。这种方式既提高了顾客体验(不必等待),又保证了数据的一致性(避免了超卖)。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/64287.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

10分钟掌握项目管理核心工具:WBS、甘特图、关键路径法全解析

一、引言 在项目管理的广阔天地里,犹如一场精心编排的交响乐演奏,每个乐器、每个音符都需精准配合才能奏响美妙乐章。而 WBS(工作分解结构)、甘特图、关键路径法无疑是这场交响乐中的关键乐章,它们从不同维度为项目管…

TCP 和 UDP 的区别:解析网络传输协议

引言 在计算机网络的世界中,TCP(Transmission Control Protocol,传输控制协议)和 UDP(User Datagram Protocol,用户数据报协议)是两种极为重要且应用广泛的传输层协议。它们在功能、特性以及适…

代码思想之快慢路径

处理业务代码的过程中,对业务代码有了一些调整,后续发现这是一种代码思想 在一段复杂的逻辑里,我把查询redis操作写在了前面, 业务逻辑: 如果需要不打压就退出本次处理 查询redis拿到商品需要打压的次数 如果次数 …

Java 溯本求源之基础(三十一)——泛型

目录 1. 泛型的定义与基本概念 2. 泛型的优势 3. 泛型的基本语法 3.1 泛型类 3.2 泛型方法 3.3 泛型接口 4. 泛型的边界 4.1 上限通配符(? extends T) 4.2 下限通配符(? super T) 5. 泛型的类型擦除 6. 泛型的使用场景…

纯 HTML+CSS+JS 实现一个炫酷的圣诞树动画特效

纯 HTMLCSSJS 实现一个炫酷的圣诞树动画特效 前言 圣诞节快到了,今天给大家带来一个简单但是效果不错的圣诞树动画特效。这个特效完全使用原生 HTML、CSS 和 JavaScript 实现,包含闪烁的星星、随机彩灯等元素,非常适合节日气氛!…

Maven:Java项目构建与管理的利器

在Java开发领域,Maven无疑是一个举足轻重的工具。它不仅简化了项目的构建和依赖管理,还促进了团队协作和持续集成。本文将深入探讨Maven的核心功能、基本配置以及在实际项目中的应用。 Maven简介 Maven是Apache基金会下的一个开源项目,旨在…

【ES6复习笔记】Promise对象详解(12)

1. 什么是 Promise? Promise 是 JavaScript 中处理异步操作的一种机制,它可以让异步操作更加容易管理和控制。Promise 对象代表一个异步操作的最终完成或失败,并提供了一种方式来处理操作的结果。 2. Promise 的基本语法 Promise 对象有三…

【RAG实战】语言模型基础

语言模型赋予了计算机理解和生成人类语言的能力。它结合了统计学原理和深度神经网络技术,通过对大量的样本数据进行复杂的概率分布分析来学习语言结构的内在模式和相关性。具体地,语言模型可根据上下文中已出现的词序列,使用概率推断来预测接…

【ES6复习笔记】Map(14)

概念 Map 是 JavaScript 中的一种数据结构,它允许你存储键值对,并且可以通过键来访问对应的值。在本教程中,我们将学习如何声明、添加、删除、获取和遍历 Map 集合。 ES6 提供了 Map 数据结构。它类似于对象,也是键值对的集合。…

富芮坤FR800X系列之PWM输出程序应用设计

文章目录 前言1.设计背景2.简介3.如何设计控制调光的接口呢4.硬件设计5.软件设计5.1.软件流程图5.2.软件代码 6.小结 前言 版权归作者所有、未经允许、请勿转载。 读者对象: 本文档主要适用以下工程师: 嵌入式系统工程师 单片机软件工程师 IOT固…

Ftrans数据摆渡系统 搭建安全便捷跨网文件传输通道

一、专业数据摆渡系统对企业的意义 专业的数据摆渡系统对企业具有重要意义,主要体现在以下几个方面‌: 1、‌数据安全性‌:数据摆渡系统通过加密传输、访问控制和审计日志等功能,确保数据在传输和存储过程中的安全性。 2、‌高…

EasyPoi 使用$fe:模板语法生成Word动态行

1 Maven 依赖 <dependency><groupId>cn.afterturn</groupId><artifactId>easypoi-spring-boot-starter</artifactId><version>4.0.0</version> </dependency> 2 application.yml spring:main:allow-bean-definition-over…

定义Shape:打造属于你的独特图形

自定义Shape:打造属于你的独特图形 在Android开发中,自定义图形绘制是一个非常重要的技能,尤其是在需要实现复杂UI或特定设计需求时。Android提供了android.graphics.drawable.shapes包,其中包含了一些基本的形状类,如RectShape、OvalShape等。然而,有时这些基本形状无法…

python 渗透开发工具之SQLMapApi Server不同IP服务启动方式处理 解决方案SqlMapApiServer外网不能访问的情况

目录 说在前面 什么是 SQLMapAPI 说明 sqlmapApi能干什么 sqlmapApi 服务安装相关 kali-sqlmap存放位置 正常启动sqlmap-api server SqlMapApi-Server 解决外网不能访问情况 说在前面 什么是sqlmap 这个在前面已经说过了&#xff0c;如果这个不知道&#xff0c;就可以…

【基础还得练】 KKT 条件

优秀教程-真正理解拉格朗日乘子法和 KKT 条件&#xff1a; link优秀教程-最优化(6)&#xff1a;一般约束优化问题的最优性理论&#xff1a; link KKT条件&#xff08;Karush-Kuhn-Tucker条件&#xff09;是非线性规划中的一组必要条件&#xff0c;在某些情况下也是最优解的充分…

使用 Webpack 优雅的构建微前端应用❕

Module Federation 通常译作“模块联邦”&#xff0c;是 Webpack 5 新引入的一种远程模块动态加载、运行技术。MF 允许我们将原本单个巨大应用按我们理想的方式拆分成多个体积更小、职责更内聚的小应用形式&#xff0c;理想情况下各个应用能够实现独立部署、独立开发(不同应用甚…

Boost之log日志使用

不讲理论&#xff0c;直接上在程序中可用代码&#xff1a; 一、引入Boost模块 开发环境&#xff1a;Visual Studio 2017 Boost库版本&#xff1a;1.68.0 安装方式&#xff1a;Nuget 安装命令&#xff1a; #只安装下面几个即可 Install-package boost -version 1.68.0 Install…

【MySQL】十四,MySQL 8.0的隐藏索引

在MySQL 8.0之前的版本中&#xff0c;索引只能直接删除。如果删除后发现引起了系统故障&#xff0c;又必须进行创建。当表的数据量比较大的时候&#xff0c;这样做的代价就会非常高。 在MySQL 8.0中&#xff0c;提供了隐藏索引。如果想删除某个索引&#xff0c;那么在实际删除…

【ES6复习笔记】解构赋值(2)

介绍 解构赋值是一种非常方便的语法&#xff0c;可以让我们更简洁地从数组和对象中提取值&#xff0c;并且可以应用于很多实际开发场景中。 1. 数组的解构赋值 数组的解构赋值是按照一定模式从数组中提取值&#xff0c;然后对变量进行赋值。下面是一个例子&#xff1a; con…

爬虫数据存储:Redis、MySQL 与 MongoDB 的对比与实践

爬虫的核心任务是从网络中提取数据&#xff0c;而存储这些数据是流程中不可或缺的一环。根据业务需求的不同&#xff0c;存储的选择可能直接影响数据处理的效率和开发体验。本文将介绍三种常用的存储工具——Redis、MySQL 和 MongoDB&#xff0c;分析它们的特点&#xff0c;并提…