java多线程复习与巩固(二)

synchronized关键字详解

      synchronized早期是重量级锁,是因为只要使用synchronized关键字都会调用操作系统内核(用户态——> 内核态)。1.6版本引入偏向锁和自旋锁,这两种类型的锁可以不调用操作系统内核 这样效率就会有所提升。与volatile不同的是 synchronized 汇编 执行的是 lock cmpxchg 保证了数据的一致性。

     java的每个对象都可以做为锁,具体标签形式有一下三种:

     对于普通方法,锁当前实例对象

     对于静态方法,锁当前类的class对象

     对于同步方法快,锁的是括号里配置的对象

     代码块同步使用monitorenter 和 monitorexit指令实现。monitorenter指令是在编译后插入到同步代码块的开始位置,而monitorexit是插入到方法结束处和异常处,JVM要保证每个monitorenter必须有对应的monitorexit与之配对。任何对象都有
一个monitor与之关联,当且一个monitor被持有后,它将处于锁定状态。线程执行到monitorenter
指令时,将会尝试获取对象所对应的monitor的所有权,即尝试获得对象的锁。

    先来看下java对象的内存结构 (在64位操作系统中 一个普通对象就是64字节?)。缓存行了解一下---

    如图是hotspot 锁对象markWord存储内容

扫描二维码关注公众号,回复: 11258589 查看本文章

如图 在分代年龄这里是4位字符 也就是说 最大为 1111 = 15 。也就是说java 最大年龄是15,垃圾回收会用到年龄

       偏向锁在Java 6和Java 7里是默认启用的,但是它在应用程序启动几秒钟之后才激活,如有必要可以使用JVM参数来关闭延迟:-XX:BiasedLockingStartupDelay=0。如果你确定应用程序里所有的锁通常情况下处于竞争状态,可以通过JVM参数关闭偏向锁:-XX:-UseBiasedLocking=false,那么程序默认会进入轻量级锁状态。

锁升级过程

简单过程

详情

开始偏向锁

不存在竞争的情况下 对象头的markword 记录线程的id 获得偏向锁。

存在竞争的情况下 后来者会发生自旋 将markword 锁记录指针指向当前线程栈的所记录,获得轻量级锁。这里栈针的数量就是锁自旋的次数。

当自旋达到次数CAS操作依然没有成功,那么就获得了重量级锁。

猜你喜欢

转载自blog.csdn.net/weixin_38700984/article/details/106171285