JDK6 synchronize锁的优化

自适应自旋锁

共享数据持有时间较短的时候,切换线程开销不值得,通过循环等待锁释放,不让出CPU资源。

锁的自旋次数不固定,由前一次在同一个锁上的自旋时间及锁的持有者状态,也就是成功率来确定自旋次数,达到自旋次数还未获取锁就进入阻塞。适应自旋锁会根据自旋获取锁的成功率来调整自旋次数,如果获取锁成功率高会调高自旋次数,否则反之。

锁消除

JIT编译时,对运行上下文扫描,去除不可能存在竞争的锁。

public void methed(String s){
    ...
    //StringBuffer是线程安全的由于sb只会在此方法使用,不可能被其他线程引用
    //因此sb属于不可能被共享的资源,JVM会自动消除内部的锁。
    StringBuffuer sb = new StringBuffer();
    sb.append(...);
    ...

}

锁粗化

将多个连续的加锁、解锁操作(锁对象相同)连接在一起,扩展成一个范围更大的锁,避免频繁反复加锁解锁。

偏向锁

大多数情况下, 锁不存在多线程竞争的情况, 都是单个线程持有。如果一个线程获得了锁,此时MarkWord的结构也变为偏向锁结构,当线程再次请求锁时,只需检查MarkWord的锁标记位是否为偏向锁以及当前线程Id是否等于Mark Word中的ThreadId即可,省去了大量有关锁申请的操作。偏向锁用于只有一个线程访问同步块或同步方法的场景。

轻量级锁

轻量级锁由偏向锁升级而来,在代码进入同步块时候,如果对象没有被锁定,虚拟机首先将在当前线程的栈帧中建立名为锁记录(lock record)的空间,并将对象头中的Mark Word复制到锁记录中,然后虚拟机将使用CAS操作尝试将对象头的Mark Word替换为指向锁记录的指针,并将线程栈帧中的Lock Record里的owner指针指向MarkWord。如果成功,那么这个线程就拥有了这个对象的锁,并且将Mark Word中的标记位改为00,表示这个对象处于轻量级锁状态。如果失败,表示有线程竞争,当前线程便尝试使用自旋锁来获取锁。如果当有两条以上的线程在抢占资源,那轻量级锁就不再有效,要膨胀为重量级锁,锁的状态更改为10。轻量级锁用于线程交替执行同步块或者同步方法的场景。


偏向锁、轻量级锁、重量级锁对比汇总

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值