Java中常见的锁类型
常见的锁分类大致有:排它锁、共享锁、乐观锁、悲观锁、分段锁、自旋锁、公平锁、非公平锁、可重入锁等。
排他锁和共享锁
synchronized
就是一个排他锁,ReentrantLock
也是一个排它锁,而ReentrantReadWriteLock
则是一个读共享锁,写排他锁
乐观锁和悲观锁
- 乐观锁和悲观锁是一种锁的思想,乐观锁对并发数据修改持乐观态度,通过CAS非加锁方式尝试修改数据,而悲观锁对数据修改持保守态度,修改都要加排它锁,也常用于和数据库隔离级别结合使用
分段锁
- 分段锁也是一种锁思想,对数据分段加锁已提高并发效率,比如jdk8之前的ConcurrentHashMap,jdk8后采用CAS+synchronized。通过hashCode计算到索引后对数据分段加锁
自旋锁
- 通过不断的轮询来尝试获取锁,是一种占用CPU时间的非阻塞锁,当锁的等待时间短,效率会很高
公平锁和非公平锁
- 同步锁按照线程申请锁的顺序,非同步则不保证。
synchronized
和ReentrantLock
都是非同步锁。ReentrantLock
可在构造方法里指定为同步锁
可重入锁
- 又叫递归锁,具体是当外部申请锁的操作获得了锁,内部申请锁的操作也会获得锁,有点类似与进程中破坏请求保持原则来预防死锁一样,要么全部申请成功,要么全部失败
锁消除
JVM会加锁的代码进行逃逸分析,当发现是单线程时,会去掉代码所加的锁,以达到优化。
synchronized是JVM层次实现的,在高并发的情况下性能不如代码层次实现的Lock高效,但是synchronized一直在被优化,现在差距已经不大了,是官方推荐的方式。