Java:synchronized关键字引出的多种锁
前言
Java 中的 synchronized
关键字可以在多线程环境下用来作为线程安全的同步锁。本文不讨论 synchronized
的具体使用,而是研究下synchronized
底层的锁机制,以及这些锁分别的优缺点。
一 synchronized机制
synchronized
关键字是JAVA中常用的同步功能,提供了简单易用的锁功能。
synchronized
有三种用法,分别为:
- 用在普通方法上,能够锁住当前对象。
- 用在静态方法上,能够锁住类
- 用在代码块上,锁住的是
synchronized
()里的对象
在JDK6之前,synchronized
使用的是重量级锁制,在之后synchronized
加入了锁膨胀机制,显著提升了synchronized
关键字的效率。
基于synchronized
关键字,我们来了解下几种类别的锁,并且讲解synchronized
的锁膨胀机制。
synchronized
锁是非公平锁。并且一个被synchronized
锁住的对象或类,就是一把锁。
另外一提,所有锁都是存储在Java对象头里的,Java对象头里的Mark Word里默认存储对象的HashCode,分代年龄和锁标记位。也就是说Mark Word记录了锁的状态
二 锁膨胀机制与几类锁
锁膨胀是不可逆的
2.1 偏向锁
synchronized
在JDK1.6以后默认开启偏向锁
,synchronized
最初都是偏向锁
表现:一个线程获取锁成功后,会在对象头里记录线程ID,以后该线程获取和释放锁都没有任何花费。(因为该锁已经被绑定在该线程上了,且在膨胀前不会改变),如果其他线程尝试获取这个锁,偏向锁
将会膨胀为轻量锁
。
优点:在只有一个线程使用锁的时候获取和退出锁没有任何花费
缺点:锁竞争激烈会很快升级为轻量锁
,那么维持偏向锁
的过程就是在浪费计算机资源。(不过因为偏向锁
本身就很轻量,因此浪费的资源并不多)
小结:只有一个线程使用锁的情况下,synchronized
使用的锁为偏向锁
。
如果锁竞争激烈,可以通过配置JDK禁用偏向锁
。
2.2 轻量锁
一把锁不止一个线程使用,则
偏向锁
膨胀为轻量锁
表现:线程获取轻量锁
时,会直接用CAS
修改对象头里锁的记录,如果修改失败,代表此时锁存在多个线程的竞争,轻量锁
将会膨胀为重量锁
。
优点:在线程之间使用锁不存在竞争时,一次CAS
操作就能获取和退出锁
缺点:与偏向锁
类似
小结:只要一把锁不止一个线程获取过,偏向锁
就会膨胀为轻量锁
。
2.3 重量锁
一把锁存在多线程竞争,则
轻量锁
开始自旋,自旋一定次数后仍没获取锁,则膨胀为重量锁
(存在竞争时,轻量锁
虽然会先自旋,但是最终往往都会膨胀为重量锁
)
表现:线程获取重量锁
时,如果获取失败(即锁已被其他线程获取),则使用自适应自旋锁
,自旋一定次数后仍没获取锁,则进入阻塞队列等待。
优点:未获取到的锁进入阻塞队列,节约CPU资源。(好吧感觉其实是没有啥优点)
缺点:重量锁
是通过对象内部的监视器(monitor)实现,其中monitor的本质是依赖于底层操作系统的Mutex Lock实现,操作系统实现线程之间的切换需要从用户态到内核态的切换,切换成本非常高。
小结:只要一把锁存在多线程竞争,轻量锁
就会膨胀为重量锁
。
自旋锁
synchronized
的轻量锁
,重量锁
,使用了自适应自旋锁
进行性能优化
首先介绍自旋锁
表现:线程获取锁失败后,不会进入阻塞等待,而是再次尝试去获取锁,如此反复,直到获取到锁,或者自旋结束那么会阻塞等待。
解决问题:在某些场景下,线程持有锁的时间非常短。在线程获取锁失败后,如果线程进入阻塞将会带来线程上下文的切换,上下文切换的时间可能反而高于线程反复尝试获取锁的时间。
此时线程原地等待去重复获取锁。反而在性能上更有优势。
缺点:
- 单核CPU没有线程并行,反复尝试会导致进程无法继续运行。
- 重复尝试导致了CPU的占用,如果CPU资源紧张的话反而会性能下降
- 如果锁的竞争时间过长,不仅没有性能提升,还浪费了大量CPU资源。
优化:使用自适应自旋锁
。自适应自旋锁会根据之前的锁获取记录,优化调整自旋时间,避免造成不必要的自旋。
三 具体synchronized流程
Java:synchronized关键字引出的多种锁的更多相关文章
- Java Synchronized 关键字
本文内容 Synchronized 关键字 示例 Synchronized 方法 内部锁(Intrinsic Locks)和 Synchronization 参考资料 下载 Demo Synchron ...
- Java synchronized 关键字详解
Java synchronized 关键字详解 前置技能点 进程和线程的概念 线程创建方式 线程的状态状态转换 线程安全的概念 synchronized 关键字的几种用法 修饰非静态成员方法 sync ...
- Java synchronized关键字用法(清晰易懂)
本篇随笔主要介绍 java 中 synchronized 关键字常用法,主要有以下四个方面: 1.实例方法同步 2.静态方法同步 3.实例方法中同步块 4.静态方法中同步块 我觉得在学习synchro ...
- java synchronized关键字浅探
synchronized 是 java 多线程编程中用于使线程之间的操作串行化的关键字.这种措施类似于数据库中使用排他锁实现并发控制,但是有所不同的是,数据库中是对数据对象加锁,而 java 则是对将 ...
- java synchronized关键字
引用其他人的一段话 Java语言的关键字,当它用来修饰一个方法或者一个代码块的时候,能够保证在同一时刻最多只有一个线程执行该段代码. 一.当两个并发线程访问同一个对象object中的这个synchro ...
- Java synchronized关键字的理解
转载自:http://blog.csdn.net/xiao__gui/article/details/8188833 在Java中,synchronized关键字是用来控制线程同步的,就是在多线程的环 ...
- java synchronized关键字浅析
synchronized这个关键字想必学Java的人都应该知道. 直接上例子: 方法级别实例 public class AtomicInteger { private int index; publi ...
- [java] java synchronized 关键字详解
Java语言的关键字,可用来给对象和方法或者代码块加锁,当它锁定一个方法或者一个代码块的时候,同一时刻最多只有一个线程执行这段代码.当两个并发线程访问同一个对象object中的这个加锁同步代码块时,一 ...
- java synchronized关键字的底层实现
每个对象都有一个锁(Monitor,监视器锁),class对象也有锁,如果synchronized关键字修饰同步代码块,通过反编译可以看到,其实是有个monitorenter和monitorexit指 ...
随机推荐
- Blend_技巧篇_导入PSD文件制作ToggleButton (Z)
原文:Blend_技巧篇_导入PSD文件制作ToggleButton (Z) 系统: Win7sp1 32位 IDE: Microsoft VisualStudio 2013 Ultimate Ble ...
- VCL to UniGUI Migration Wizard
Free Evaluation Edition of The Automatic Migration Scripting Wizard For Converting Legacy Delphi Cod ...
- JQuery采纳CSS实现DOM显示和隐藏要素
今天参加了Code Review活动.阅读编写代码的同事,感到满足当前功能的实现,但是从长远来看,,无论角度还是从代码重用是来看显然不佳维修点. 有什么需要看的权利.通过选择不同的选项下拉框需求,为了 ...
- Win32 键盘事件 - 击键消息、字符消息、插入符号(光标)
注:以下内容为学习笔记,多数是从书本.资料中得来,只为加深印象,及日后参考.然而本人表达能力较差,写的不好.因非翻译.非转载,只好选原创,但多数乃摘抄,实为惭愧.但若能帮助一二访客,幸甚! 以下内容主 ...
- Easyui Tab刷新
Easyui Tab刷新: function refreshTab(title){ var tab = $('#id').tab('getTab',title); $('#id').tab('upda ...
- SGI STL中内存池的实现
最近这两天研究了一下SGI STL中的内存池, 网上对于这一块的讲解很多, 但是要么讲的不完整, 要么讲的不够简单(至少对于我这样的初学者来讲是这样的...), 所以接下来我将把我对于对于SGI ST ...
- WPF 动态模拟CPU 使用率曲线图
原文:WPF 动态模拟CPU 使用率曲线图 在工作中经常会遇到需要将一组数据绘制成曲线图的情况,最简单的方法是将数据导入Excel,然后使用绘图功能手动生成曲线图.但是如果基础数据频繁更改, ...
- Mvvm Light Toolkit 入门
原文:Mvvm Light Toolkit 入门 前言 之前学习UWP的时候就一直看到有关MVVM的资料但是一直没有系统的去学,最近正好有时间,特地来攻破这个点,顺便学习一下VS与GitHub的链接和 ...
- WPF/Silverlight 页面绑定Model验证机制升级版
关于WPF/Silverlight的数据验证,想必大家都不陌生了. 各大牛的博客里都不泛对这方面讨论的文章. 个人比较赞赏 JV9的“Silverlight实例教程”里的Validation数据验证. ...
- Windows Phone SplashScreen初始屏幕示例
protected override void OnLaunched(LaunchActivatedEventArgs args) { if (args.PreviousExecutionState ...