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

乐观锁与悲观锁

乐观锁


        乐观锁是一种并发控制的机制,其核心思想是假设多个事务之间的冲突是不太可能发生的,因此在事务处理之前不会加锁,而是在事务提交的时候再检查是否有冲突。如果发现冲突,就会回滚事务,重新尝试。

实现乐观锁的方式

1. 版本号机制:
  • 每个数据记录都关联一个版本号,当读取数据时,将版本号一同读出。在更新数据时,只有当版本号匹配时才能执行更新操作,否则认为是冲突,需要进行回滚或其他处理。
  • 适用于数据库表中的记录,常用于数据库乐观锁实现。
2. 时间戳(Timestamp)机制:
  • 每个事务执行时都记录一个时间戳,更新数据时带上时间戳。当提交时,检查时间戳,如果发现其他事务已经更新了数据,就认为发生了冲突。
  • 可以在数据库中记录事务的开始时间作为时间戳,也可以使用递增的整数作为版本号。
3. CAS(Compare and Swap)操作:
  • 使用原子性的CAS操作来判断是否发生冲突。在Java中,Atomic 类提供了一些原子操作,如 compareAndSet
  • 适用于基于内存的数据结构,如Java中的AtomicIntegerAtomicLong等。

乐观锁的优点

  1. 性能好: 在低并发环境下,乐观锁的性能通常优于悲观锁,因为不需要额外的加锁和解锁操作。

  2. 无阻塞: 由于乐观锁不会一开始就阻塞线程,因此适用于读操作较频繁、写操作较少的场景。

乐观锁的缺点

  1. 冲突处理: 当多个事务发生冲突时,需要进行冲突处理,通常是通过回滚事务,重新尝试。

  2. 不适用于高并发写操作: 当写操作较频繁时,乐观锁的性能可能下降,因为不断的冲突会导致事务的回滚和重试。

  3. 无法解决所有并发问题: 乐观锁机制不能解决所有并发问题,特别是在一些复杂的业务场景中。

使用乐观锁的示例(基于版本号机制):

class Account {private String accountId;private double balance;private long version; // 版本号// 省略其他代码// 更新余额的方法public void updateBalance(double amount) {// 模拟乐观锁检查if (version != getAccountVersionFromDatabase()) {throw new OptimisticLockException("Concurrent modification detected");}// 更新余额this.balance += amount;// 更新版本号version++;// 更新数据库中的版本号和余额updateAccountInDatabase();}
}

   version 是账户对象的版本号,每次更新时都需要检查数据库中的版本号是否一致,如果不一致,则抛出乐观锁异常。

悲观锁


        悲观锁是一种并发控制的机制,它的核心思想是在操作数据之前,悲观地认为会有并发操作的冲突,因此先进行加锁,确保每个时刻只有一个事务可以访问或修改共享资源。这种锁定机制确保了数据的一致性,但也可能导致性能的下降,因为多个事务可能需要等待锁的释放。

        悲观锁的实现方式主要包括数据库锁、行级锁、表级锁等,以及编程语言级别的锁,如Java中的synchronized关键字、数据库中的SELECT ... FOR UPDATE等。

实现悲观锁的方式

1. 数据库锁:
  • 行级锁(Row-level lock): 在数据库中锁定表中的某一行,确保只有一个事务可以修改这一行的数据。例如,在SQL中可以使用FOR UPDATE语句。
  • 表级锁(Table-level lock): 锁定整个表,防止其他事务访问该表中的任何数据。
2. 编程语言级别的锁:
  • 在编程语言中,通过关键字实现锁机制。例如,Java中的synchronized关键字用于同步方法或代码块,确保在同一时刻只有一个线程可以访问被锁定的资源。
3. 互斥量(Mutex):
  • 在操作系统级别,可以使用互斥量确保同一时刻只有一个线程可以访问共享资源。

悲观锁的优点

  1. 数据一致性: 悲观锁确保了数据的一致性,因为在操作数据之前先获取了锁,避免了并发冲突。

  2. 简单直观: 实现相对简单,理解容易。

悲观锁的缺点

  1. 性能开销: 悲观锁的加锁操作会带来性能开销,尤其是在高并发的情况下,因为其他事务需要等待锁的释放。

  2. 死锁风险: 当多个事务相互等待对方释放锁时,可能发生死锁。

  3. 资源争用: 多个事务争用同一个资源时,可能导致大量的等待时间,降低系统的吞吐量。

使用悲观锁的示例:

在Java中,使用synchronized关键字可以实现悲观锁:

public class BankAccount {private double balance;// 同步方法,使用悲观锁public synchronized void deposit(double amount) {balance += amount;}// 同步代码块,使用悲观锁public void withdraw(double amount) {synchronized (this) {if (balance >= amount) {balance -= amount;} else {System.out.println("Insufficient funds");}}}
}

   synchronized关键字确保在同一时刻只有一个线程可以执行depositwithdraw方法。这就是一种悲观锁的实现方式。

相关文章:

乐观锁与悲观锁

乐观锁 乐观锁是一种并发控制的机制,其核心思想是假设多个事务之间的冲突是不太可能发生的,因此在事务处理之前不会加锁,而是在事务提交的时候再检查是否有冲突。如果发现冲突,就会回滚事务,重新尝试。 实现乐观锁的方…...

【算法】堆排序

算法-堆排序 前置知识 堆(即将更新) 思路 我们现在有一个序列,怎么对它排序? 这是一个非常经典的问题,这里我们使用一个借助数据结构的算法——堆排序解决。 这里有一个序列,要对它升序排序 4 7 3 6 5 …...

51单片机应用从零开始(三)

51单片机应用从零开始(一)-CSDN博客 51单片机应用从零开始(二)-CSDN博客 详解 KEIL C51 软件的使用建立工程-CSDN博客 详解 KEIL C51 软件的使用设置工程编绎与连接程序-CSDN博客 目录 1. 用单片机控制第一个灯亮 2. 认识单片…...

如何在 Nginx Proxy Manager(NPM)上部署静态网站

前言 众所周知,我们在之前介绍过 Nginx Proxy Manager(以下简称 NPM) 这个反向代理的神器,对于一些 Docker 搭建的 Web 项目,NPM 能够很轻松地给他们做反向代理。 然而对于一些静态网站,小伙伴们可能不知道怎么用 NP…...

http的几种方法

http的几种方法在 rfc2616 中进行了定义: https://www.rfc-editor.org/rfc/rfc2616.html#page-51 HEAD方法:HEAD方法和GET方法相同,只不过服务端只返回头,不返回消息体。GET方法:用于获取资源POST方法:用于…...

var、let、const关键字的特性,以及let、const暂时性死区的作用

var、let和const都是JavaScript中的关键字,用于声明变量。 var关键字声明的变量是函数作用域或全局作用域的,它在整个函数或全局范围内都是可用的。var没有块级作用域。 let关键字声明的变量是块级作用域的,它只在包含它的代码块中可用。le…...

IDEA 高分辨率卡顿优化

VM设置优化 -Dsun.java2d.uiScale.enabledfalse 增加该条设置,关闭高分切换 https://intellij-support.jetbrains.com/hc/en-us/articles/115001260010-Troubleshooting-IDE-scaling-DPI-issues-on-Windows​intellij-support.jetbrains.com/hc/en-us/articles/1…...

【AIGC】一起学习prompt提示词(4/4)【经典】【15种提示词技巧】

写的时候并没有设计好,要做多少期,还是有始有终的比较好,为了方便阅读,我把之前的3期,改下名字,放到这里。 【AIGC】一起学习prompt提示词(1/4) 内容摘要:提示词是什么…...

Linux实战一天一个小指令--《文件管理/文件查找》

阿丹: 作为一个java程序员进行实战开发不接触linux操作系统基本上是不可能的,所以这个专题就出现了,本文章重点解决大家关于文件管理以及文件查找查看的疑惑。我将采用语法基础用法并在下面进行高级语法的总结使用,方便大家学习和…...

CocosCreator3.8神秘面纱 CocosCreator 项目结构说明及编辑器的简单使用

我们通过Dashboard 创建一个2d项目,来演示CocosCreator 的项目结构。 等待创建完成后,会得到以下项目工程: 一、assets文件夹 assets文件夹:为资源目录,用来存储所有的本地资源,如各种图片,脚本…...

JJJ:python学习笔记

p4 没有编译的过程 源码和输入得到输出 静态语言:编译型 脚本语言:解释型 p5 又叫做胶水语言 p7 p8 p10...

SpringSecurity6从入门到上天系列第七篇:讲明白SpringBoot的自动装配完善上篇文章中的结论

文章目录 一:SpringBoot的自动装配 1:从run方法到入口类内容被注册到注解解读器中。 2:解析入口类注解到加载Bean实例 大神链接:作者有幸结识技术大神孙哥为好友,获益匪浅。现在把孙哥视频分享给大家。 孙哥链接&am…...

ClickHouse 原理解析之基础知识总结

ClickHouse 基础知识整理 参考ClickHouse 官方文档:https://clickhouse.com/docs/en/intro 一:行式存储和列式存储 1.行式存储和列式存储的区别 1.1 概念说明 行式存储:指存储结构化数据时,在底层的存储介质上,数据是以行的方式来组织的,即存储完一条记录的所有字段,再…...

最小花费——最短路

在 n 个人中,某些人的银行账号之间可以互相转账。这些人之间转账的手续费各不相同。给定这些人之间转账时需要从转账金额里扣除百分之几的手续费,请问 A 最少需要多少钱使得转账后 B 收到 100 元。 输入格式 第一行输入两个正整数 n,m,分别表…...

Spark DataFrame join后移除重复的列

在Spark&#xff0c;两个DataFrame做join操作后&#xff0c;会出现重复的列。例如&#xff1a; Dataset<Row> moviesWithRating moviesDF.join(averageRatingMoviesDF,moviesDF.col("movieId").equalTo(averageRatingMoviesDF.col("movieId")));其s…...

NextJS工程部署到阿里云linux Ecs

nextjs项目有多种部署方式&#xff0c;本文介绍最简单的一种方式&#xff0c;将源码上传到云服务器&#xff0c;编译后使用pm2后台运行nextjs工程。 检查node、npm是否安装 查看npm版本&#xff0c;如果版本较低先升级npm版本 npm -v卸载 yum remove nodejs npm -y安装新版…...

汽车以太网IOP测试新利器

IOP测试目的 汽车以太网物理层IOP&#xff08;Interoperability &#xff09;测试&#xff0c;即测试被测对象以太网物理层之间的互操作性。用于验证车载以太网PHY能否在有限时间内建立稳定的链路&#xff1b;此外&#xff0c;还用于验证车载以太网PHY可靠性相关的诊断特性&am…...

高防IP是什么?如何隐藏源站IP?如何进行防护?

高防IP是针对互联网服务器遭受大流量的DDoS攻击后导致服务不可用的情况下,推出的付费增值服务。用户在数据不转移的情况下,就可以通过配置高防IP , 将攻击流量引流到高防|P,确保源站的稳定可靠。高防IP采用的技术手段包括DDoS防护、WAF ( Web应用程序防火墙)等,它能够有效抵御来…...

ElasticSearch---查询es集群状态、分片、索引

查看es集群状态&#xff1a; curl -XGET http://localhost:9200/_cat/health?v如果?后面加上pretty&#xff0c;能让返回的json格式化。 加上?v的返回结果&#xff0c;如下&#xff1a; epoch timestamp cluster status node.total node.data shards pri rel…...

Angular 使用教程——基本语法和双向数据绑定

Angular 是一个应用设计框架与开发平台&#xff0c;旨在创建高效而精致的单页面应用 Angular 是一个基于 TypeScript 构建的开发平台。它包括&#xff1a;一个基于组件的框架&#xff0c;用于构建可伸缩的 Web 应用&#xff0c;一组完美集成的库&#xff0c;涵盖各种功能&…...

SEO网站广告如何与本地化营销相结合

SEO网站广告与本地化营销的结合&#xff1a;如何提升本地企业的市场竞争力 在当今数字化经济的浪潮中&#xff0c;SEO网站广告和本地化营销已经成为企业营销的两大重要手段。如何将这两者有机地结合&#xff0c;以实现最大的营销效益&#xff0c;是许多企业面临的重要课题。本…...

Windows系统优化终极指南:用Win11Debloat免费快速提升性能

Windows系统优化终极指南&#xff1a;用Win11Debloat免费快速提升性能 【免费下载链接】Win11Debloat A simple, lightweight PowerShell script that allows you to remove pre-installed apps, disable telemetry, as well as perform various other changes to declutter an…...

DYOR 嘉创地产 02421.HK

文章目录1.公司概况1.1 简介1.2 股权结构1.3 核心资质与定位2.业务布局3.财务与市场表现&#xff1a;业绩承压&#xff0c;规模迷你3.1 业绩大幅下滑3.2 市场表现落后3.3 规模在行业中垫底4.核心优势5.潜在风险与隐忧6.小结参考文献1.公司概况 1.1 简介 嘉创地产是一家脱胎于…...

手把手教你用PyTorch复现YOLOv8的Pose Head:从零搭建关键点检测模块

手把手教你用PyTorch复现YOLOv8的Pose Head&#xff1a;从零搭建关键点检测模块 在计算机视觉领域&#xff0c;目标检测与姿态估计的结合正成为工业界和学术界的热点。YOLOv8作为YOLO系列的最新成员&#xff0c;其姿态估计模块&#xff08;Pose Head&#xff09;的设计尤为精妙…...

Pixel Couplet Gen基础教程:Streamlit+ModelScope零配置环境搭建步骤详解

Pixel Couplet Gen基础教程&#xff1a;StreamlitModelScope零配置环境搭建步骤详解 1. 项目介绍与准备 Pixel Couplet Gen是一款融合了传统春节文化与现代像素艺术风格的AI春联生成器。它基于ModelScope大模型驱动&#xff0c;通过Streamlit构建了独特的8-bit复古游戏界面&a…...

【原创改进代码】考虑电动汽车移动储能特性的多区域电网功率波动平抑优化调控附python代码

✅作者简介&#xff1a;热爱科研的Matlab仿真开发者&#xff0c;擅长毕业设计辅导、数学建模、数据处理、建模仿真、程序设计、完整代码获取、论文复现及科研仿真。 &#x1f34e; 往期回顾关注个人主页&#xff1a;Matlab科研工作室 &#x1f447; 关注我领取海量matlab电子…...

终极指南:如何让2012-2015年老款Mac安装最新macOS系统

终极指南&#xff1a;如何让2012-2015年老款Mac安装最新macOS系统 【免费下载链接】OpenCore-Legacy-Patcher Experience macOS just like before 项目地址: https://gitcode.com/GitHub_Trending/op/OpenCore-Legacy-Patcher 您的2012-2015年老款Mac是否已被苹果官方抛…...

SEO和SEM对于中小企业的意义是什么_SEO 和 SEM 的报告指标有哪些

SEO和SEM对于中小企业的意义是什么 在当今的数字化时代&#xff0c;中小企业如何在竞争激烈的市场中脱颖而出&#xff0c;已成为每一个企业家关注的焦点。搜索引擎优化&#xff08;SEO&#xff09;和搜索引擎营销&#xff08;SEM&#xff09;作为两种重要的数字营销手段&#…...

管理员命令提示符 命令提示符 cmd

命令提示符区别...

开源新形态:从代码到Prompt的转变

【导语&#xff1a;3月末&#xff0c;开源作者yetone发布新项目voice-input-src&#xff0c;以独特方式“开源”&#xff0c;即用自然语言Prompt生成代码&#xff0c;此做法引发讨论&#xff0c;或预示开源模式新转变。】AI驱动的语音输入法开源项目开源作者yetone在GitHub上发…...