多线程锁机制面试
目录
乐观锁的底层原理
ReentrantLock的实现原理
读写锁 ReentrantReadWriteLock
synchronized 底层原理
Lock和synchronized的区别
乐观锁的底层原理
-
版本号机制 在数据库表中添加一个版本号字段(如 version),每次更新数据时都会将版本号加1。 当尝试更新数据时,会检查当前版本号是否与开始操作时获取的版本号一致。 如果版本号一致,则更新成功;如果不一致,则表示数据已被其他事务修改,更新失败。
-
时间戳机制 类似于版本号机制,但在表中使用时间戳字段来记录数据的最后修改时间。 更新数据时,同样需要检查当前时间戳是否与开始操作时获取的时间戳一致。 如果时间戳一致,则更新成功;如果不一致,则表示数据已被其他事务修改,更新失败。
-
CAS 操作 CAS (Compare and Swap) 是一种无锁算法,用于原子地更新一个变量的值。 在多线程环境中,CAS 操作可以用来实现乐观锁。 CAS 操作通常由 CPU 提供硬件级别的支持,也可以通过软件模拟实现。
乐观锁的优点 减少了锁的竞争,提高了系统的吞吐量。 适用于读多写少的场景。
乐观锁的缺点 如果数据更新频繁,可能导致更新失败的次数增多,从而降低性能。 需要额外的字段存储版本号或时间戳
ReentrantLock的实现原理
ReentrantLock 是 Java 并发包(java.util.concurrent.locks)中提供的一个可重入互斥锁,它提供了比 synchronized 关键字更灵活的锁机制。其核心实现原理基于 AbstractQueuedSynchronizer(AQS)框架
实现原理: 可重入性:ReentrantLock 允许同一个线程多次获取锁而不被阻塞,通过内部维护一个计数器来记录锁被当前线程获取的次数。每次成功获取锁,计数器加1;每次释放锁,计数器减1,当计数器为0时,其他线程才能获取到锁。
公平性与非公平性: 公平锁(FairSync):按照请求锁的顺序来分配锁,先来的线程优先获取锁。实现这一点需要维持一个先进先出(FIFO)的等待队列,线程在尝试获取锁失败后会加入队列末尾等待。
非公平锁(NonfairSync):允许插队,即当前线程尝试获取锁时,即使有其他线程已经在等待队列中,只要锁当前是空闲的,就允许立即获取。这可以减少线程的上下文切换,提高吞吐量,但可能牺牲公平性。
自旋锁与AQS:ReentrantLock 在尝试获取锁时可能会使用自旋策略,即线程在无法立即获取锁时不是立即挂起而是循环等待一段时间,这尤其适用于锁持有时间很短的场景。AQS通过维护一个双向链表来管理等待线程,链表中的每个节点代表一个等待状态的线程,通过CAS操作来保证状态更新的原子性。
条件变量:ReentrantLock 还支持条件变量(Condition),允许线程在满足特定条件前等待,其他线程可以唤醒这些等待的线程。每个条件变量都有自己的等待队列。 锁的释放:释放锁时,不仅会减少锁的持有计数,还会唤醒等待队列中的下一个线程(如果有的话),这通常通过LockSupport类的park/unpark方法实现。
读写锁 ReentrantReadWriteLock
-
读写锁的基本概念 读锁: 多个读操作可以同时进行,不会互相阻塞。
-
写锁: 写操作是独占的,在写操作进行时不允许任何读操作。
-
ReentrantReadWriteLock 的特点 可重入性: 同一个线程可以多次获取同一个锁。 公平性: 可以选择公平锁或非公平锁。 读写分离: 读操作和写操作分别使用不同的锁。
-
ReentrantReadWriteLock 的使用 创建锁: 通过 ReentrantReadWriteLock 类创建读写锁实例。 获取锁: 通过 readLock() 和 writeLock() 方法分别获取读锁和写锁。 锁定与解锁: 使用 lock() 和 unlock() 方法来锁定和解锁。
synchronized 底层原理
synchronized 关键字在 Java 中用于实现线程间的互斥和同步,它基于 JVM 层面的监视器锁(Monitor)机制来实现。以下是 synchronized 的底层原理概述:
1.对象头(Object Header): Java 对象在内存中有一个对象头,其中包含了对象的元数据,如哈希码、GC代信息以及锁信息。 锁信息存储在一个称为 Monitor 的结构中,这个结构是 JVM 提供的,用来实现同步。
Monitor 监视器: 当一个线程试图进入一个由 synchronized 修饰的方法或代码块时,它会尝试获取该对象的 Monitor 锁。 如果锁是可用的,线程将进入临界区,并且锁状态会变为“已锁定”。 如果锁已经被其他线程持有,请求锁的线程将被阻塞,直到锁被释放。
字节码指令: 在字节码级别,synchronized 关键字会被转换为 monitorenter 和 monitorexit 指令。 monitorenter 指令会在执行前尝试获取锁,而 monitorexit 指令则在执行后释放锁。 这些指令操作的是对象头中的 Monitor 结构。
锁升级: 为了提高性能,JVM 实现了锁的升级机制。初始时,锁可能处于无锁状态,然后根据竞争情况升级为偏向锁、轻量级锁或重量级锁。 偏向锁适用于没有多个线程竞争的情况,轻量级锁使用 CAS 操作尝试获取锁,而重量级锁则涉及到操作系统级别的线程阻塞和唤醒。 即时编译优化: HotSpot VM 会对 synchronized 块进行优化,例如,如果检测到锁没有竞争,它可以避免使用重量级锁,从而减少上下文切换和线程调度的开销。
Lock和synchronized的区别
-
语法糖 vs 显式控制 synchronized: 是 Java 的关键字,提供了语法级别的支持,用于声明同步代码块或同步方法。 Lock: 是 java.util.concurrent.locks 包中的一个接口,需要显式地获取和释放锁。
-
锁的获取与释放 synchronized: 在进入同步代码块或方法时自动获取锁,并在退出时自动释放锁。 Lock: 需要显式地调用 lock() 方法获取锁,并且必须在不再需要锁时显式地调用 unlock() 方法释放锁。为了防止因异常而导致锁未被释放,通常将 unlock() 方法放在 finally 块中。
-
可重入性 synchronized: 具有可重入性,允许同一个线程多次获取同一个锁。 Lock: 也支持可重入性,但需要通过 ReentrantLock 类实现。
-
公平性 synchronized: 默认是非公平锁。 Lock: 可以通过 ReentrantLock 类的构造函数指定锁是否公平。
-
中断响应 synchronized: 无法响应中断请求,即使线程被中断,仍然持有锁。 Lock: 支持中断请求,可以通过 tryLock 方法尝试获取锁,并且可以监听中断信号。
-
锁状态的检查 synchronized: 无法检查锁是否被当前线程持有。 Lock: 可以通过 isHeldByCurrentThread() 方法检查当前线程是否持有锁。
-
等待/通知机制 synchronized: 直接使用 wait() 和 notify() 方法来实现线程间的等待/通知。 Lock: 通过 Condition 接口实现等待/通知机制,需要与 Lock 结合使用。
-
异常处理 synchronized: 在发生异常时会自动释放锁,因此不会出现死锁。 Lock: 发生异常时,如果没有显式地释放锁,可能会导致死锁。
相关文章:
多线程锁机制面试
目录 乐观锁的底层原理 ReentrantLock的实现原理 读写锁 ReentrantReadWriteLock synchronized 底层原理 Lock和synchronized的区别 乐观锁的底层原理 版本号机制 在数据库表中添加一个版本号字段(如 version),每次更新数据时都会将版本号…...
《SQL 中计算地理坐标两点间距离的魔法》
在当今数字化的世界中,地理数据的处理和分析变得越来越重要。当我们面对一个包含地理坐标数据的表时,经常会遇到需要计算两点之间距离的需求。无论是在物流配送路线规划、地理信息系统应用,还是在基于位置的服务开发中,准确计算两…...
微服务可用性设计
一、隔离 对系统或资源进行分割,实现当系统发生故障时能限定传播范围和影响范围。进一步的,通过隔离能够降低系统之间得耦合度,使得系统更容易维护和扩展。某些业务场景下合理使用隔离技巧也能提高整个业务的性能。我理解隔离本质就是一种解…...
【扒代码】dave readme文档翻译
jerpelhan/DAVE (github.com) 摘要 低样本计数器估算选定类别对象的数量,即使在图像中只有少量或没有标注样本的情况下。目前最先进的技术通过对象位置密度图的总和来估算总数量,但这种方法无法提供单个对象的位置和大小,这对于许多应用来说…...
c语言---文件
这一节我准备分三个部分来带领大家了解文件 ——一、有关文件的基础知识 ————二、文件的简单操作 ————————三、文件结束的判定 ————————————四、文件缓冲区 一、文件的基础知识: 首先在了解文件之前,我们需要了解C/C程序内存…...
Windows系统下Go安装与使用
step1: 下载go语言SDK 下载地址:https://go.dev/dl/ 下载后选择合适位置安装即可,我选择D盘 在安装完成后,可以通过go env 命令检测是否安装成功。在“命令提示符”界面输入“go env”命令,如果显示如下类似结果则说明…...
day24-测试之接口测试基础
目录 一、接口的定义 二、接口的优点 三、API接口 四、接口测试流程 五、网络基础概念 六、HTTP和RURL 七、get和post请求 八、数据格式 九、状态码 十、restful风格 十一、接口工具 一、接口的定义 程序之间协作所要遵循的一套规范、标准 二、接口的优点 2.1.责任…...
TSN 交换机
TSN(Time-Sensitive Networking)交换机是一种支持时间敏感网络协议的网络交换设备,用于在以太网网络中实现低延迟、高确定性的数据传输。TSN 是一组 IEEE 802 标准的集合,旨在通过标准化的方式,将传统的以太网扩展到需…...
针对thinkphp站点的漏洞挖掘和经验分享
0x1 前言 浅谈 目前在学习和研究thinkphp相关漏洞的打法,然后最近对于thinkphp资产的收集方面有了一个简单的认识,然后写一篇新手看的thinkphp相关的漏洞收集和挖掘的文章来分享下。然后后面是给师傅们分享下后台文件上传,然后直接打一个ge…...
MySQL数据库入门,pycharm连接数据库—详细讲解
一.安装MySQL 1.常用MySQL5.7,首先安装MySQL, (一) (二) (三) (四) (五) 2.配置环境变量 打开MySQL安装路径,在其中找到…...
.bat文件快速运行vue项目
如何使用bat文件快速运行vue项目? 新建个文件,改名为serve.bat。 在文件中写入以下内容: # cd 项目路径 cd D:\projects\xxx npm run serve pausecd 项目所在的路径 npm run dev/serve ,取决于项目的启动方法,打…...
数据结构(邓俊辉)学习笔记】优先级队列 07——堆排序
1.算法 作为完全二叉堆的一个应用,这节来介绍堆排序算法。 是的,谈到优先级队列,我们很自然地就会联想到排序。因为就其功能而言,包括完全二叉堆在内的任何一种优先级队列都天生地具有选取功能,也就是选取其中的最大…...
npm install pnpm -g 报错的解决方法
npm install pnpm -g 报错的解决方法 npm error code ETIMEDOUT npm error errno ETIMEDOUT npm error network request to https://registry.npmjs.org/pnpm failed, reason: npm error network This is a problem related to network connectivity. npm error network In mo…...
集师知识付费小程序开发
智慧生活,从选择一款优质知识付费小程序起航 在这个信息爆炸的时代,知识成为了最宝贵的财富。我们渴望不断学习,提升自我,追求更高品质的生活。而一款优质的知识付费小程序,就如同照亮前行道路的明灯。 它是知识的宝库…...
前端开发提效工具——用户自定义代码片段
做开发总是会有大量的代码要写,但是有时候某些代码是非常基础但是很多,我们就可以把这一部分整合起来,使用一个很简短的关键字来快速唤出。 如何新建这样的代码段? 1.在VSCode当中找到Snippets,然后点击 2.之后会弹出…...
docker容器安全加固参考建议——筑梦之路
这里主要是rootless的方案。 在以 root 用户身份运行 Docker 会带来一些潜在的危害和安全风险,这些风险包括: 容器逃逸:如果一个容器以 root 权限运行,并且它包含了漏洞或者被攻击者滥用,那么攻击者可能会成功逃出容器…...
基于 Appium 的 App 爬取实战
除了运行 Appium 的基本条件外,还要一个日志输出库 安装: pip install loguru 思路分析 首先我们观察一下整个 app5 的交互流程,其首页分条显示了电影数据, 每个电影条目都包括封面,标题, 类别和评分 4…...
nvm与node安装
参考: 一文搞定NVM安装所有问题NVM UI解决nodejs下载慢问题 node_mirror: http://npmmirror.com/mirrors/node/ npm_mirror: http://registry.npmmirror.com/mirrors/npm/解决nvm list available报错问题 Could not retrieve https://npm.taobao.org/mirrors/node/…...
【电子通识】什么是MSL湿敏等级
潮敏失效是塑料封装表贴器件在高温焊接工艺中表现出来的特殊的失效现象。 造成此类问题的原因是器件内部的潮气膨胀后使得器件发生损坏。 MSL是“Moisture Sensitivity Level(湿气敏感性等级)”的缩写,针对需进行回流焊的产品设定了MSL基准。…...
【ARM 芯片 安全与攻击 5.4 -- Meltdown 攻击与防御介绍】
文章目录 什么是 Meltdown 攻击?Meltdown 攻击的基本原理Meltdown 攻击代码示例Meltdown 攻击在芯片中的应用应用场景Meltdown 攻击与瞬态攻击、测信道攻击的关系针对 Meltdown 攻击的防御硬件级防御Summary什么是 Meltdown 攻击? Meltdown 攻击是一种利用处理器乱序执行(o…...
4.【Python】Python3 注释
第一步:分析与整理 注释1. 注释的作用 不影响程序执行,只提高可读性。帮助理解代码逻辑,方便团队协作。2. 单行注释 以 # 开头,直到行末的所有内容均为注释。 # 这是一个注释 print("Hello, World!") # 这也是注释3. 多…...
OpenGL 调试方式
调试手段总览 API 级错误检查:glGetError、断言、包装宏调试输出机制:GL_KHR_debug、glDebugMessageCallback、QOpenGLDebugLogger着色器与程序调试:编译/链接日志、离线编译器、颜色编码调试渲染结果调试:FBO 检查、glReadPixels…...
从Simulink的Vector信号到C代码数组:手把手拆解初始化(initialize)与步进(step)函数的生成逻辑
从Simulink的Vector信号到C代码数组:手把手拆解初始化与步进函数的生成逻辑 在嵌入式系统开发中,Simulink模型到C代码的转换过程往往被视为一个"黑箱"——工程师们习惯性地点击生成按钮,然后接受输出的代码文件。然而,当…...
CircuitPython内存优化:冻结模块原理与嵌入式开发实践
1. 项目概述:当微控制器项目撞上内存墙在嵌入式开发的世界里,尤其是玩转像Adafruit Circuit Playground Express这类资源受限的微控制器时,我们常常会与一个无形的“天花板”迎头相撞——内存限制。你可能正兴致勃勃地为你的智能徽章或互动艺…...
chipKIT平台与PIC32开发板:32位MCU的Arduino兼容方案
1. Arduino兼容的chipKIT平台与PIC32开发板概述在嵌入式开发领域,32位微控制器(MCU)正逐步取代传统的8位MCU,成为创客、学生和专业工程师的首选。Microchip Technology公司推出的chipKIT平台,正是这一趋势下的产物。chipKIT平台基于高性能的3…...
开源安全工具openclaw-killer:Nginx Lua环境威胁检测与防护实践
1. 项目概述:一个开源安全工具的诞生与使命最近在安全研究圈子里,一个名为openclaw-killer的项目引起了我的注意。这个由nkzprod维护的开源工具,名字就透着一股“杀气”——“OpenClaw杀手”。乍一看,你可能会以为这是某个游戏外挂…...
学妹问降完AI重复率反涨10个点怎么办?这款降AI工具同时降AI率重复率
学妹问降完AI重复率反涨10个点怎么办?这款降AI工具同时降AI率重复率 学妹凌晨发来的紧急求助 3 月 24 号凌晨 1:17 学妹发来消息:「学姐我刚送知网测——AI 率从 65% 降到 9% 过了!但重复率从 18% 涨到 28% 不达标了!这怎么办」…...
告别“找不同”游戏:STANet的时空注意力如何让AI看懂遥感图中的建筑变化?
告别“找不同”游戏:STANet如何让AI像人类一样理解遥感图中的建筑变化? 想象一下,你面前有两张相隔数月的卫星照片,需要找出其中新建的楼盘或消失的农田。传统方法就像玩“找不同”游戏——逐像素对比颜色差异,稍有不慎…...
掌握TA-Lib Python技术分析库:从入门到精通的完整指南 [特殊字符]
掌握TA-Lib Python技术分析库:从入门到精通的完整指南 🚀 【免费下载链接】ta-lib-python Python wrapper for TA-Lib (http://ta-lib.org/). 项目地址: https://gitcode.com/gh_mirrors/ta/ta-lib-python TA-Lib Python技术分析库是金融量化交易…...
告别烦人黑窗口!QT Creator控制台程序输出完美嵌入IDE的两种方法
告别烦人黑窗口!QT Creator控制台程序输出完美嵌入IDE的两种方法 每次调试QT控制台程序时,那个突然弹出的黑窗口是否总让你手指悬停在AltTab键上?作为深耕QT开发多年的技术顾问,我见过太多开发者被这个"窗口刺客"打断思…...
