关于Java 锁的知识整理与回顾(我的笔记):java
锁有哪些,分别用来干吗?函数
Java实现锁有两种方式,synchronized关键字和Lock性能
(1)Lock(可判断锁状态)优化
Lock是基于JDK层面实现。Lock的实现主要有ReentrantLock、ReadLock和WriteLock(引出锁分类:)spa
①乐观锁/悲观锁:操作系统
乐观锁认为读多写少,乐观的认为拿数据时,不会改数据,因此不会上锁,而在更新数据时才会判断有无数据更新。悲观锁悲观的认为,写多,拿数据时先设定数据被修改了,每次在读写数据时都会上锁。线程
②公平锁/非公平锁:code
ReentrantLock在构造函数中提供是否公平锁的初始化方式(默认是非公平锁,就是说能够变成公平锁。即他和synchronized不一样之处之一):对象
public ReentrantLock() { sync = new NonfairSync(); }
public ReentrantLock(boolean fair) { sync = fair ? new FairSync() : new NonfairSync(); }
非公平锁不按套路出牌,有可能形成“饥饿”现象,可是它的实际执行效率、吞吐量要高于公平锁。blog
③独享锁/共享锁
独享锁是指该锁一次只能被一个线程所持有 (ReentrantLock、 Synchronized),共享锁反之(ReadWriteLock)。
④互斥锁/读写锁(ReadWriteLock)
互斥锁/读写锁像是独享锁/共享锁的具体的实现。(从字面意思就解释很清楚了)
⑤可重入锁
指同一个线程在外层方法获取锁的时候,在进入内层方法会自动获取锁(ReentrantLock和Synchronized)。可重入锁可必定程度避免死锁。
⑥自旋锁
自旋锁是指尝试获取锁的线程不会阻塞,而是采用循环的方式尝试获取锁。“采用循环的方式”说明要一直占用CPU资源,浪费啊。在判断持有锁的线程会在很短的时间释放锁,用它就划算了,不用上下文来回切换了。
⑦偏向锁/轻量级锁/重量级锁
这是jdk1.6中对Synchronized锁作的优化,对象头在不一样锁状态下的标志位存储。
偏向锁:加锁/解锁开销少,适用于只有一个线程访问同步块场景;轻量级锁:竞争线程不会堵塞,追求响应速度时用它;重量级锁:线程竞争不使用自旋,不会消耗CPU,吞吐量大。(尽可能先考虑使用轻量级锁)
(2)synchronized(不可判断锁状态):
synchronized基于JVM层面实现。synchronized能够把任意的非Null的对象看成琐,它是非公平锁(且没法变成公平锁)、独享的可重入的悲观锁。使用synchronized比较省事儿,可是它是一个重量级操做,须要调用操做系统相关接口,性能较低,锁开销可能较大。
同时,synchronized它在不断的被优化,在JDK1.5以后synchronized引入了偏向锁,轻量级锁和重量级锁,java 6有适应自旋、锁粗化、偏向锁等,以后还设置一堆标记位,减小得到锁和释放锁带来的性能消耗,锁开销小了,效率高了。