一、Synchronized基本特点
- 开始是轻量级锁,如果锁被持有时间很长变为重量级锁
- 实现轻量级锁的时候大概率使用自旋锁(CAS实现)
- 开始是乐观锁,如果锁冲突概率变多变为悲观锁
- Synchronized是一个非公平锁
- Synchronized不是读写锁
- Synchronized是可重入锁
二、Synchronized锁升级
无锁:
顾名思义,不加锁
偏向锁:
第一个尝试加锁的线程优先进入偏向锁状态,偏向锁并不是真正的加锁,而是在对象头中做一个偏向锁标记,记录这个锁是属于哪个线程的。如果后序没有线程竞争这个锁,就啥事没有,0开销。如果后面有别的线程来竞争这个锁(比对刚才保存的对象头当中的记录判断是不是之前的线程)就取消原来的偏向锁状态,进入一般的轻量级锁的状态。
偏向锁本质就是延迟加锁,能不加就不加,没办法了就加,但是对象头当中的标记要做好。
轻量级锁:
随着别的线程竞争锁,锁已经变成了轻量级锁(自适应的自旋锁),此处的轻量级锁就是CAS实现的,它不涉及用户态内核态的转换,不涉及线程的阻塞和调度,只是多用了一点CUP。
重量级锁:
自旋锁一直空转也不行,因此这里的自旋锁不会一直自旋,在旋转10次之后就会升级为重量级锁,这里的重量级锁就是内核提供的mutex
- 执行加锁操作要先进入内核态。
- 在内核态判定锁是否被占用。
- 没有则加锁成功并返回用户态。
- 被占用则阻塞等待,等待被唤醒。
- 最终等到持有这个锁的线程释放锁,操作系统唤醒刚才阻塞等待的线程,尝试重新获取锁。