偏向锁浅析

  偏向锁不像自旋锁、读写锁或者synchronize修饰词这样的同步,它其实是JVM内置的一种锁机制,自JDK1.6后默认启用。换句话说,这种锁不是咱程序员能用代码来瞎操心的,JVM自己会去操心的。真想要瞎操心,就得改JVM的启动参数:

启用参数: 
-XX:+UseBiasedLocking
关闭延迟: -XX:BiasedLockingStartupDelay=0 禁用参数: -XX:-UseBiasedLocking

  既然无需我们操心,那么了解一下也是好的。偏向锁偏向于第一个获得它的线程,如果在接下来的执行过程中,该锁没有被其他的线程获取,那么持有偏向锁的线程无需再进行同步。很明显,当锁的竞争情况很少出现时,偏向锁就能提高性能,因为它比轻量级锁(如自旋锁)少了一步:CAS。

  偏向锁的加锁和解锁有点像可重入锁,它都得先知道取得锁的线程是谁,拿到锁的身份证(线程ID),下次相同的线程来了,啥都别说了,直接走快速通道pass。但如果锁的竞争比较激烈,那么偏向锁并无太大用处。我们来看看,在自旋锁和synchronize同步方法两种情况下,偏向锁的实际时延比较,这里用的是jdk1.8版本。

  一、自旋锁:代码参见自旋锁浅析的testSpinLock方法

  1、默认耗时:

count值:100000, 耗时:25毫秒.

  2、开启偏向锁,启动默认五秒之后生效:-XX:+UseBiasedLocking

count值:100000, 耗时:32毫秒.

  3、开启偏向锁,立即生效:-XX:+UseBiasedLocking -XX:BiasedLockingStartupDelay=0

count值:100000, 耗时:102毫秒.

  4、关闭偏向锁:-XX:-UseBiasedLocking

count值:100000, 耗时:30毫秒.

  我们可以看到,默认时延是最少的,因为JVM会自动优化,而无时延开启偏向锁是最高的,不开启偏向锁和开启但需要5秒启动(这时线程早跑完了,跟不开启差不多)跟默认时延差不多。

  二、synchronize同步方法:代码参见读写锁浅析的TestMyReadWriteLock方法,需要增加全局变量:

private long startTime = 0L;

  在before方法开始处加入该变量的赋值:

startTime = System.currentTimeMillis();

  再加上after方法:

    @After
    public void after()
    {
        System.out.printf("耗时:%d毫秒.\n", System.currentTimeMillis() - startTime);
    }

  1、默认耗时:

耗时:1076毫秒.

  2、开启偏向锁

耗时:1090毫秒.

  3、开启偏向锁,立即生效

耗时:1099毫秒.

   4、关闭偏向锁

耗时:1078毫秒.

   以上对比发现,这偏向锁开不开都差不多。

posted on 2019-01-28 19:34  不想下火车的人  阅读(6664)  评论(0编辑  收藏  举报

导航