synchronized在jdk1.6后做的优化:
锁消除:在synchronized修饰的代码中,如果存在临界资源的情况下,即使写了synchronized也无效,不会触发;
锁膨胀:如果一个循环中,频繁的获取和释放锁资源,这样会带来很大的资源,锁膨胀就是将锁的范围扩大,避免频繁的竞争和释放锁资源带来不必要的消耗,例如当一个synchronized锁放在for循环中时,可以将synchronized的代码块抽取出来,放到for循环体外面,这样就避免了每次循环都要去竞争和释放锁资源;
锁升级:Reetrantlock的实现是先基于乐观锁CAS尝试获取锁资源,如果拿不到才会挂起线程,synchronized在jdk1.6之前,完全就是获取不到锁就立即挂起线程,所以synchronized的性能较差,synchronized在jdk1.6做了锁升级优化,有几种以下锁的状态:
- 无锁和匿名偏向锁:当前对象没有作为锁的存在
- 偏向锁:如果当前锁资源只有一个线程频繁释放和获取锁,那么这个线程过来只需要判断当前指向线程是否是当前线程,是的话直接获取锁资源,不是的话就基于CAS方式尝试将偏向锁指向当前线程,如果获取不到就升级为轻量级锁
- 轻量级锁:会采用自旋的方式去频繁的以CAS形式获取锁资源,成功就拿到,如果自旋一定次数还没拿到,就会升级为重量级锁
- 重量级锁:拿不到锁资源就挂起