前言

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轻量锁重量锁,使用了自适应自旋锁进行性能优化

首先介绍自旋锁

表现:线程获取锁失败后,不会进入阻塞等待,而是再次尝试去获取锁,如此反复,直到获取到锁,或者自旋结束那么会阻塞等待。

解决问题:在某些场景下,线程持有锁的时间非常短。在线程获取锁失败后,如果线程进入阻塞将会带来线程上下文的切换,上下文切换的时间可能反而高于线程反复尝试获取锁的时间。
此时线程原地等待去重复获取锁。反而在性能上更有优势。

缺点:

  1. 单核CPU没有线程并行,反复尝试会导致进程无法继续运行。
  2. 重复尝试导致了CPU的占用,如果CPU资源紧张的话反而会性能下降
  3. 如果锁的竞争时间过长,不仅没有性能提升,还浪费了大量CPU资源。

优化:使用自适应自旋锁。自适应自旋锁会根据之前的锁获取记录,优化调整自旋时间,避免造成不必要的自旋。

三 具体synchronized流程

Java:synchronized关键字引出的多种锁的更多相关文章

  1. Java Synchronized 关键字

    本文内容 Synchronized 关键字 示例 Synchronized 方法 内部锁(Intrinsic Locks)和 Synchronization 参考资料 下载 Demo Synchron ...

  2. Java synchronized 关键字详解

    Java synchronized 关键字详解 前置技能点 进程和线程的概念 线程创建方式 线程的状态状态转换 线程安全的概念 synchronized 关键字的几种用法 修饰非静态成员方法 sync ...

  3. Java synchronized关键字用法(清晰易懂)

    本篇随笔主要介绍 java 中 synchronized 关键字常用法,主要有以下四个方面: 1.实例方法同步 2.静态方法同步 3.实例方法中同步块 4.静态方法中同步块 我觉得在学习synchro ...

  4. java synchronized关键字浅探

    synchronized 是 java 多线程编程中用于使线程之间的操作串行化的关键字.这种措施类似于数据库中使用排他锁实现并发控制,但是有所不同的是,数据库中是对数据对象加锁,而 java 则是对将 ...

  5. java synchronized关键字

    引用其他人的一段话 Java语言的关键字,当它用来修饰一个方法或者一个代码块的时候,能够保证在同一时刻最多只有一个线程执行该段代码. 一.当两个并发线程访问同一个对象object中的这个synchro ...

  6. Java synchronized关键字的理解

    转载自:http://blog.csdn.net/xiao__gui/article/details/8188833 在Java中,synchronized关键字是用来控制线程同步的,就是在多线程的环 ...

  7. java synchronized关键字浅析

    synchronized这个关键字想必学Java的人都应该知道. 直接上例子: 方法级别实例 public class AtomicInteger { private int index; publi ...

  8. [java] java synchronized 关键字详解

    Java语言的关键字,可用来给对象和方法或者代码块加锁,当它锁定一个方法或者一个代码块的时候,同一时刻最多只有一个线程执行这段代码.当两个并发线程访问同一个对象object中的这个加锁同步代码块时,一 ...

  9. java synchronized关键字的底层实现

    每个对象都有一个锁(Monitor,监视器锁),class对象也有锁,如果synchronized关键字修饰同步代码块,通过反编译可以看到,其实是有个monitorenter和monitorexit指 ...

随机推荐

  1. Linux性能测试 strace命令

    1  功能说明 strace 命令是一种强大的工具 ,  能够显示任何由用户空间程式发出的系统调用 .  strace 显示这些调用的参数并返回符号形式的值 .  strace 从内核接收信息 ,  ...

  2. 倒计时的CountDownTimer

    直接看这里吧,我仅仅是搬运工.  定时运行在一段时候后停止的倒计时,在倒计时运行过程中会在固定间隔时间得到通知(译者:触发onTick方法),以下的样例显示在一个文本框中显示一个30s倒计时: , 1 ...

  3. 卷积(convolution)与相关(correlation)(matlab 实现)

    1. 卷积(convolution) 输出 y(n) 是作为在 x(k) 和 h(n−k)(反转和移位)重叠之下的样本和求出的. 考虑下面两个序列: x(n)=[3,11,7,0,−1,4,2],−3 ...

  4. 算法 Tricks(六)—— 判断一个数是否为完全平方数

    int(sqrt(n)) * int(sqrt(n)) == n ? 1:0; matlab 下判断一个数是否能开方的判断是: floor(sqrt(m))^2 == m

  5. .net reactor 学习系列(二)---.net reactor界面各功能说明

    原文:.net reactor 学习系列(二)---.net reactor界面各功能说明         安装了.net reactor之后,可以在安装目录下找到帮助文档REACTOR_HELP.c ...

  6. WPF 动态绑定listview的列内容

    Binding binding = new Binding(); binding.Path = new PropertyPath("State"); listViewState.D ...

  7. [原译]WPF绘制圆角多边形

    原文:[原译]WPF绘制圆角多边形 介绍 最近,我发现我需要个圆角多边形.而且是需要在运行时从用户界面来绘制.WPF有多边形.但是不支持圆角.我搜索了一下.也没找到可行的现成例子.于是就自己做吧.本文 ...

  8. 关于WPF的ComboBox中Items太多而导致加载过慢的问题

    原文:关于WPF的ComboBox中Items太多而导致加载过慢的问题 [WFP疑难]关于WPF的ComboBox中Items太多而导致加载过慢的问题                          ...

  9. Tinyhttpd for Windows(学习型的项目,才500多行代码)

    前言 TinyHTTPd是一个开源的简易学习型的HTTP服务器,项目主页在:http://tinyhttpd.sourceforge.net/,源代码下载:https://sourceforge.ne ...

  10. MugLife静态照片变3D动画算法研究

    原文:MugLife静态照片变3D动画算法研究 MugLife app是一款可以将静态照片变成3D动画的手机应用,如下效果图所示: 大家可以看到,这个静态图具有了类3D的动画特效,是不是很好玩? 这种 ...