Java高并发程序设计(八)--提高锁性能的建议和JVM对锁的优化

一、提高锁性能的建议

(1)减小锁持有时间:

public synchronized void demo()
    {
        otherthing();
        work();
        otherthing();
    }

对于上面这种,可以只加锁给需要加锁的部分,改进如下:

public void demo()
    {
        otherthing();
        synchronized(this)
        {
            work();
        }
        otherthing();
    }

(2)减小锁粒度:

如ConcurrentHashMap,把Map分成Segment,再给每个Segment加锁,详情见原来的博客。

(3)读写分离锁代替独占锁:

使用读写锁是减小锁粒度的一种特殊方法,上面的Segment是对数据的划分,而读写锁就是对功能点的划分。

如ReentrantReadWriteLock,可以使读读不同步,适用读多写少的场景。

(4)锁分离:

如上一篇博客提到过的LinkedBlockingQueue队列,put和take在队头队尾有两把不同的锁。

(5)锁粗化:

频繁地获取锁,释放锁也是很费资源的事情,如:

public  void demo()
    {
        for(int i=0;i<100;i++)
        {
            synchronized(this)
            {
                work();
            }
        }
    }

它要获取整整一百次锁,可以改造如下:

public  void demo()
    {
        synchronized(this)
        {
            for(int i=0;i<100;i++)
            {
                work();
            }
        }
    }

 

二、JVM优化锁做出的努力

(1)锁偏向:

一个线程获得锁时,进入锁偏向模式,当它再次进入,不需要任何同步操作。对锁竞争小的时候效率很高。

可以通过虚拟机调参,-XX:+UseBiasedLocking开启

(2)轻量级锁:

锁偏向失败后,为了减小重量级锁的性能消耗,不再申请互斥量,而是在对象头中,部分字节通过CAS的方式指向线程栈中的LockRecord,如果成功则获得锁。失败膨胀成重量级锁。

(3)自旋锁:

膨胀之后,虚拟机做最后努力,假设也许只要一会就能申请到锁,于是做几个空循环,若干循环后如果得到锁,进入临界区,否则挂起。

(4)锁消除:

在JIT编译时,通过对上下文扫描,将所有完全没有竞争的资源的锁消除。比如单线程使用Vector。

posted @ 2018-08-03 15:42  蒋曾  阅读(458)  评论(0编辑  收藏  举报