前言
高效并发是从JDK1.5到JDK1.6的一个重要改进,Java开发者团队在这个版本上花费了大量的精力去实现各种锁优化技术,如适应性自旋、锁消除、锁粗化、轻量级锁和偏向锁等。这些技术都是为了在线程之间更高效地共享数据,以及解决竞争问题,从而提高程序的执行效率。
自旋锁与自适应自旋
互斥同步中,对性能最大的影响是阻塞的实现:挂起线程和恢复线程的操作都需要转入内核态去完成,这些操作给系统的并发性带来了很大的压力。为此我们设置了自旋锁,让线程执行一个忙循环(等待)。
自旋锁的评估
自旋锁本身虽然避免了线程切换的开销,但它是要占用处理器的时间。因此,如果锁被占有的时间很短,自旋等待的效果会非常好;反之,如果锁被占有的时间很长,那么自旋的线程只会白白浪费处理器资源。因此,自旋等待的时间必须有一定的限度,如果超过了限定的次数仍然没有成功获得锁,就应当使用传统的方式挂起线程。自旋次数的默认值为10,用户可以通过使用参数-XX:PreBlockSpin
来更改。
自适应自旋锁
自适应意味着自旋的时间不再固定,而是由前一次在同一个锁的自旋时间及锁的拥有者的状态来决定。如果在同一对象上,自旋等待刚刚成功获得锁,并且持有锁的线程正在运行中,那么虚拟机就会认为这次自旋也很有可能成功,进而它允许自旋等待持续相对较长时间。如果对于某个锁,自旋很少成功,那么在以后获取这个锁时将可能省略掉自旋过程,以避免浪费处理器资源。
锁消除
锁消除是指虚拟机即时编译器运行时,对一些代码要求同步,但是被检测到不可能存在共享数据竞争的锁进行消除。