锁的优化

目录

锁的四种状态

  • 从低到高,只能升级不能降级
  • 无锁状态
  • 偏向锁状态
  • 轻量级锁状态
  • 重量级锁状态

自旋锁和自适应自旋

理解

  • 同步互斥的时候会造成线程阻塞,而挂起线程和恢复线程需要转入内核态中完成
  • 有时候往往共享数据的锁定状态只会持续很短一段时间,那么如果此时将互斥的线程挂起,等待下一次获得锁再恢复线程,这样的话效率不高
  • 那么可以不讲线程立即挂起,而是让他自旋(循环)等待获得锁(这就是自旋锁),那么这样可以避免在一段很短的等待时间挂起和恢复线程的开销
  • 也就说,不立刻将线程挂起,稍微自旋等一下,这样的话或许可以减少线程挂起和恢复的开销

分析

  • 虽然可以避免切换线程的开销,但是因为在自旋,这同样是消耗cpu
  • 倘若锁被占用的时间很短,那么这短时间的自旋比起切换线程来说是值得的
  • 但是倘若锁被占用的时间很长,那么自旋会消耗cpu资源,却没有做有用的事

锁消除

理解

  • 虚拟机即使编译器在运行的时候,同步代码被检测不存在共享数据竞争,那么锁会被消除。也就是说如果数据不会被其他线程访问到,那么我们没必要加锁

锁粗化

理解

  • 我们加锁的时候会尽量减小锁的粒度,减少竞争
  • 而如果我们对同一个对象反复加锁和解锁,会造成性能损耗。
  • 所以不如把锁的粒度加大

使用

StringBuffer sb=new StringBuffer();
sb.append(s1);
sb.append(s2);
...
sb.sppend(sn);
  • 我们知道StringBuffer的append方法是加锁的,而这里的锁就是sb对象,如果我们执行每一个append方法都加锁解锁,显然这样性能是不高的,所以当虚拟机检测到一系列操作都是对同一个对象加锁,他会将同步的范围扩大,比如这里扩大到s1-sn,那么这样的话就只需加一次锁

MarkWord与线程之间的操作过程

  • java对象头中的Mark Word默认存储对象的Hashcode、分代年龄和锁标志位
  • 进入同步代码块的时候,当前线程的栈帧会建立一个名为锁记录的空间,用于存储锁对象MarkWord的拷贝

偏向锁

理解

  • 锁会偏向于第一个获得他的线程,如果后面锁没有被其他线程获取,那么持有偏向锁的线程永远不需要再进行同步

使用

  • 锁对象第一次被线程获取的时候,对象头的标志位会设为偏向模式,同时通过CAS操作把这个线程的ID记录在MarkWord中
  • 如果CAS操作成功,持有偏向锁的线程以后每一进入这个锁的同步块时,不需要进行同步操作
  • 当另一个线程尝试获取这个锁,偏向锁结束

轻量级锁

理解

  • 传统的重量级锁需要使用系统互斥量来实现,而轻量级锁的意图是在没有多线程竞争的情况下,减少重量级锁的系统互斥量产生的性能消耗

使用

  • 虚拟机使用CAS操作尝试将MarkWord更新为指向LockRecord的指针,如果更新成功表示线程就拥有该对象的锁
  • 如果失败,会检查MarkWord是否指向当前线程的栈帧,如果是表示当前线程已经拥有这个锁,如果不是说明这个锁被其他线程抢占,此时膨胀为重量级锁

轻量级锁和偏向锁

  • 重量级锁需要使用操作系统的互斥量(常常使用一个整型量,0表示解锁,而其他所有的值则表示加锁。通过互斥量使同一资源同时只允许一个访问者对其进行访问)来实现
  • 轻量级锁依据大部分的锁在同步周期内不存在竞争,使用CAS操作避免使用互斥量的开销
  • 偏向锁的则是在无竞争的情况下,连CAS操作都不做

我觉得分享是一种精神,分享是我的乐趣所在,不是说我觉得我讲得一定是对的,我讲得可能很多是不对的,但是我希望我讲的东西是我人生的体验和思考,是给很多人反思,也许给你一秒钟、半秒钟,哪怕说一句话有点道理,引发自己内心的感触,这就是我最大的价值。(这是我喜欢的一句话,也是我写博客的初衷)

作者:jiajun 出处: http://www.cnblogs.com/-new/

本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。如果觉得还有帮助的话,可以点一下右下角的【推荐】,希望能够持续的为大家带来好的技术文章!想跟我一起进步么?那就【关注】我吧。

jvm系列 (三) ---锁的优化的更多相关文章

  1. JVM系列三:JVM参数设置

    JVM系列三:JVM参数设置.分析   不管是YGC还是Full GC,GC过程中都会对导致程序运行中中断,正确的选择不同的GC策略,调整JVM.GC的参数,可以极大的减少由于GC工作,而导致的程序运 ...

  2. JVM系列三:JVM参数设置、分析

    不管是YGC还是Full GC,GC过程中都会对导致程序运行中中断,正确的选择不同的GC策略,调整JVM.GC的参数,可以极大的减少由于GC工作,而导致的程序运行中断方面的问题,进而适当的提高Java ...

  3. JVM系列三:JVM参数设置、分析(转载)

    不管是YGC还是Full GC,GC过程中都会对导致程序运行中中断,正确的选择不同的GC策略,调整JVM.GC的参数,可以极大的减少由于GC工作,而导致的程序运行中断方面的问题,进而适当的提高Java ...

  4. 【转载】JVM系列三:JVM参数设置、分析

    不管是YGC还是Full GC,GC过程中都会对导致程序运行中中断,正确的选择不同的GC策略,调整JVM.GC的参数,可以极大的减少由于GC工作,而导致的程序运行中断方面的问题,进而适当的提高Java ...

  5. [转]JVM系列三:JVM参数设置、分析

    不管是YGC还是Full GC,GC过程中都会对导致程序运行中中断,正确的选择不同的GC策略,调整JVM.GC的参数,可以极大的减少由于GC工作,而导致的程序运行中断方面的问题,进而适当的提高Java ...

  6. 转:JVM系列三:JVM参数设置、分析

    转自:http://www.cnblogs.com/redcreen/archive/2011/05/04/2037057.html 不管是YGC还是Full GC,GC过程中都会对导致程序运行中中断 ...

  7. MySQL系列(三)--数据库结构优化

    良好的数据库逻辑设计和物理设计是数据库高性能的基础,所以对于数据库结构优化是很有必要的 数据库结构优化目的: 1.减少数据的冗余 2.尽量避免在数据插入.删除和更新异常 例如:有一张设计不得当的学生选 ...

  8. JVM系列三(垃圾收集器).

    一.概述 1. 哪些内存需要回收 上篇文章 我们介绍了 Java 内存运行时区域的各个部分,其中程序计数器.虚拟机栈.本地方法栈三个区域随线程而生,随线程而灭,在这几个区域内就不需要过多考虑回收的问题 ...

  9. jvm系列三垃圾回收

    三.垃圾回收 1.如何判断对象可以回收 引用计数法 弊端:循环引用时,两个对象的计数都为1,导致两个对象都无法被释放 可达性分析算法 JVM中的垃圾回收器通过可达性分析来探索所有存活的对象 扫描堆中的 ...

随机推荐

  1. 无法启动 IIS Express Web 服务器

    问题描述:我用的是vs2015,有时候打开自己的项目,点击调试运行,会失败,弹出窗口,告诉我,无法启动 IIS Express Web 服务器,我就纳闷了,刚才还好好,怎么这会就不行了,各种试,都不行 ...

  2. Sublime Text中安装插件来实现px与rem间的换算

    今天在群里无意中看到了群友分享的一篇关于移动端的文章.里面其他内容我倒不大感兴趣,反而是rem让我提起了兴趣. 首先来谈一下rem,rem是CSS3中新增加的一个单位值,它和em单位一样,都是一个相对 ...

  3. php之试触法----error--关键字的误用

    实际开发中,在不同网页的输出中,常常有许多公共的代码或者变量需要使用,于是定义了以下类来缩减代码量 如下代码所示: <?php class universalClass { function w ...

  4. Dubbo Data length too large: 11557050, max payload: 8388608 传输数据超限

    com.alibaba.dubbo.remoting.transport.AbstractCodec.checkPayload() ERROR Data length too large: 11557 ...

  5. 玩玩微信公众号Java版之二:接收、处理及返回微信消息

    前面已经配置了微信服务器,那么先开始最简单的接收微信消息吧~   可以用我们的微信号来直接进行测试,下面先看测试效果图:   这是最基本的文本消息的接收.处理及返回,来看看是怎么实现的吧!   首先可 ...

  6. USB的包结构及包分类

    USB的传输总是低位在前,高位在后. USB的传输方向:从设备到主机的数据为输入:从主机到设备的数据叫做输出. 1. 包结构 以同步域开始,紧跟着一个包标识符PID(Packet Identifier ...

  7. Sql for Oracle基本操作关键字

    Sql for Oracle基本操作关键字 +SQL TOP子句 TOP 子句用于规定要返回的记录的数目 SELECT column_name(s) FROM table_name WHERE ROW ...

  8. oracle-sql入门练习及答案

    1,通过命令行方式打开sqlplus2,执行登录命令sqlplus scott/scott@192.168.248.129/orcl3进行sqlplus命令测试Set time onSet pages ...

  9. GetConsoleScreenBufferInfo 函数--获取控制台屏幕缓冲区信息

    GetConsoleScreenBufferInfo函数 来源:https://msdn.microsoft.com/en-us/library/ms683171(v=vs.85).aspx 作用 获 ...

  10. Mybitis配置文件中的别名以及mapper中的namespace

    1 基本知识 MyBatis中如果每次配置类名都要写全称也太不友好了,我们可以通过在主配置文件中配置别名,就不再需要指定完整的包名了. 别名的基本用法: <configuration> & ...