Java关键字synchronized原理与锁的状态实例分析
学习知识要善于思考,思考,再思考!今天golang学习网小编就给大家带来《Java关键字synchronized原理与锁的状态实例分析》,以下内容主要包含等知识点,如果你正在学习或准备学习文章,就都不要错过本文啦~让我们一起来看看吧,能帮助到你就更好了!
一、Java中锁的概念
自旋锁:是指当一个线程获取锁的时候,如果锁已经被其它线程获取,那么该线程将循环等待,然后不断的判断锁是否能被成功获取,直到获取到锁才会退出循环。
乐观锁:假定没有冲突,在修改数据时如果发现数据和之前获取的不一致,则读最新数据,重试修改。
悲观锁:假定会发生并发冲突,同步所有对数据的相关操作,从读数据就开始上锁。
独享锁(写):给资源加上写锁,线程可以修改资源,其它线程不能再加锁(单写)。
共享锁(读):给资源加上读锁后只能读不能修改,其它线程也只能加读锁,不能加写锁(多度)。看成Semaphore(信号量)理解即可。
可重入锁&不可重入锁:线程拿到一把锁之后,可以自由进入同一把锁所同步的其它代码。
公平锁&非公平锁:争抢锁的顺序,如果是按先来后到,则为公平。即能保证抢锁的顺序和抢到锁的顺序一致则为公平锁。
二、同步关键字synchronized特性
特性:可重入、独享、悲观锁。
锁相关的优化:
锁消除 :开启锁消除的参数有
-XX:+DoEscapeAnalysis、-XX:+EliminateLocks。锁粗化:JDK做了锁粗化的优化,但我们自己可从代码层面优化。
1、锁消除示例
/**
* 锁消除示例,JIT即时编译,进行了锁消除
* @author 刘亚楼
* @date 2020/1/16
*/
public class LockEliminationExample {
/**
* StringBuilder线程不安全,StringBuffer用了synchronized关键字,是线程安全的
* 针对下面这种单线程加锁、解锁操作,JIT会进行优化,进行锁消除
*/
public static void eliminateLock() {
StringBuffer stringBuffer = new StringBuffer();
stringBuffer.append("a");
stringBuffer.append("b");
stringBuffer.append("c");
stringBuffer.append("a");
stringBuffer.append("b");
stringBuffer.append("c");
stringBuffer.append("a");
stringBuffer.append("b");
stringBuffer.append("c");
}
}2、锁粗化示例
/**
* 锁粗化示例
* @author 刘亚楼
* @date 2020/1/16
*/
public class LockCoarseningExample {
/**
* 针对下面这种无意义的加锁操作,JIT会进行优化,对变量i的所有操作放到一个同步代码块里
*/
public static void lockCoarsening() {
int i = 0;
synchronized (LockCoarseningExample.class) {
i++;
}
synchronized (LockCoarseningExample.class) {
i--;
}
synchronized (LockCoarseningExample.class) {
i++;
}
synchronized (LockCoarseningExample.class) {
i++;
i--;
i++;
}
}
}备注:锁消除和锁粗化的区别在于锁消除是针对单个线程重复加解锁做的优化,最终没有锁的存在。而锁粗化不只是针对单线程,且最终还是有锁的存在。
三、synchronized关键字原理
1、关于Mark Word
首先,对象在堆中由对象头、实例数据和对齐填充组成。
对象头包含两部分信息,第一部分用于存储对象自身的运行时数据,如哈希码、GC分代年龄、锁状态标志、线程持有的锁、偏向锁id等,这部分数据官方称为"Mark Word"。
对象头的另一部分是类型指针,即对象指向它的类元数据的指针,虚拟机通过这个指针来确定这个对象是哪个类的实例。
synchronized实现的锁是通过改变对象头的"Mark Word"来实现的。
"Mard Word"在32位和64位的虚拟机(未开启压缩指针)中分别为32位和64位。32位虚拟机"Mark Word"如下:

2、锁的状态变化
(1) 无锁 → 轻量级锁
无锁变成轻量级锁时,多个线程会读取对象的对象头的无锁状态mark word内容,然后进行cas操作进行修改,预期值是无锁状态mark word内容,新值是轻量级锁状态mark word内容,若修改成功,Lock record address指向成功获取锁的线程的Lock Record。
演示流程如下:

(2) 轻量级锁 → 重量级锁
由于未成功获取锁的线程会自旋,长时间自旋会消耗CPU资源,因此自旋到一定次数会进行锁升级,由轻量级锁转变为重量级锁。
重量级锁是通过object monitor(对象监视器)实现的,对象监视器包括entryList(锁池)、owner(持锁者)、waitSet(等待集合)等。
升级为重量级锁时对象头mark word的内容是monitor address(对象监视器地址),指向对象监视器。
演示流程如下:

备注:抢锁失败线程会进入entryList(锁池),在调用wait方法后,线程会进入waitSet(等待集合),waitSet中的线程被唤醒后会重新进入entryList。
(3) 关于偏向锁
加锁之后不解锁,针对单线程
所谓偏向就是偏心,单线程加锁之后就不再解锁,减少了加锁→业务处理→释放锁→加锁操作流程。
在JDK6以后,默认已经开启了偏向锁这个优化,通过JVM参数-XX:-UseBiasedLocking来禁用偏向锁,若偏向锁开启,只有一个线程抢锁,可获取到偏向锁。
关于偏向锁Mark Word内容如下:

偏向标记第一次有用,出现过争用后就没用了。
偏向锁本质就是无锁,如果没有发生过任何多线程争抢锁的情况,JVM认为就是单线程,无需做同步。
备注:JVM为了少干活,同步在JVM底层是有很多操作来实现的,如果没有争用,就不需要去做同步操作。
(4) 完整的锁升级过程
如果未开启偏向锁,无锁状态会先升级为轻量级锁,轻量级锁自选到一定程度升级为重量级锁。
如果开启了偏向锁,有两种情况:
当锁未被占用时,会升级为无锁,无锁再升级为轻量级锁,再由轻量级锁升级为重量级锁。
当锁被占用时,会升级为轻量级锁,再由轻量级锁升级到重量级锁。

文中关于java,synchronized的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《Java关键字synchronized原理与锁的状态实例分析》文章吧,也可关注golang学习网公众号了解相关技术文章。
欧阳明高院士力挺插电混动技术:全面超越,未来已来!
- 上一篇
- 欧阳明高院士力挺插电混动技术:全面超越,未来已来!
- 下一篇
- win10教育版过期了怎么激活激活方法
-
- 文章 · java教程 | 1分钟前 |
- Java自定义异常提升代码可读性技巧
- 367浏览 收藏
-
- 文章 · java教程 | 12分钟前 |
- Eclipse配置JavaWeb项目运行步骤
- 155浏览 收藏
-
- 文章 · java教程 | 24分钟前 |
- Java如何绘制学生成绩趋势图
- 430浏览 收藏
-
- 文章 · java教程 | 27分钟前 |
- JavaStreamreduce操作详解与使用技巧
- 293浏览 收藏
-
- 文章 · java教程 | 27分钟前 |
- 消息队列幂等处理技巧解析
- 215浏览 收藏
-
- 文章 · java教程 | 34分钟前 |
- throws与throw区别详解及使用场景
- 435浏览 收藏
-
- 文章 · java教程 | 36分钟前 |
- JavaJDK17安装配置详解
- 144浏览 收藏
-
- 文章 · java教程 | 42分钟前 |
- 动态枚举映射静态成员的实现方法
- 238浏览 收藏
-
- 文章 · java教程 | 57分钟前 |
- Java简易投票系统可视化实现教程
- 469浏览 收藏
-
- 文章 · java教程 | 1小时前 |
- Java集合size方法的优缺点分析
- 500浏览 收藏
-
- 文章 · java教程 | 1小时前 |
- JavaPaths.get路径使用全解析
- 465浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
-
- ChatExcel酷表
- ChatExcel酷表是由北京大学团队打造的Excel聊天机器人,用自然语言操控表格,简化数据处理,告别繁琐操作,提升工作效率!适用于学生、上班族及政府人员。
- 3210次使用
-
- Any绘本
- 探索Any绘本(anypicturebook.com/zh),一款开源免费的AI绘本创作工具,基于Google Gemini与Flux AI模型,让您轻松创作个性化绘本。适用于家庭、教育、创作等多种场景,零门槛,高自由度,技术透明,本地可控。
- 3424次使用
-
- 可赞AI
- 可赞AI,AI驱动的办公可视化智能工具,助您轻松实现文本与可视化元素高效转化。无论是智能文档生成、多格式文本解析,还是一键生成专业图表、脑图、知识卡片,可赞AI都能让信息处理更清晰高效。覆盖数据汇报、会议纪要、内容营销等全场景,大幅提升办公效率,降低专业门槛,是您提升工作效率的得力助手。
- 3453次使用
-
- 星月写作
- 星月写作是国内首款聚焦中文网络小说创作的AI辅助工具,解决网文作者从构思到变现的全流程痛点。AI扫榜、专属模板、全链路适配,助力新人快速上手,资深作者效率倍增。
- 4561次使用
-
- MagicLight
- MagicLight.ai是全球首款叙事驱动型AI动画视频创作平台,专注于解决从故事想法到完整动画的全流程痛点。它通过自研AI模型,保障角色、风格、场景高度一致性,让零动画经验者也能高效产出专业级叙事内容。广泛适用于独立创作者、动画工作室、教育机构及企业营销,助您轻松实现创意落地与商业化。
- 3831次使用
-
- 提升Java功能开发效率的有力工具:微服务架构
- 2023-10-06 501浏览
-
- 掌握Java海康SDK二次开发的必备技巧
- 2023-10-01 501浏览
-
- 如何使用java实现桶排序算法
- 2023-10-03 501浏览
-
- Java开发实战经验:如何优化开发逻辑
- 2023-10-31 501浏览
-
- 如何使用Java中的Math.max()方法比较两个数的大小?
- 2023-11-18 501浏览

