1. 概况

  • ReentrantLock 是可重入的,也就是说如果当前线程拥有这把锁,进入到下一个加了同样锁的方法中时,不用等待直接获取,只是对象头字段增加了一个标记,标记进入了两次。

  • 比 Synchronized 更加灵活,多一个尝试获取锁的办法,和获取锁超时时间

2. 类定义

  1. public class ReentrantLock implements Lock, java.io.Serializable {
  2. }

ReentrantLock 实现 Lock 接口,Lock 接口包含包含以下几个方法:

  • lock(),获取锁,当前线程未获取成功时会进入等待状态
  • lockInterruptibly(),可中断的获取锁
  • boolean tryLock(),尝试获取锁,如果当前线程没有获取到锁,会立马返回 false
  • boolean tryLock(long time, TimeUnit unit) throws InterruptedException,尝试获取锁,指定时间还没有获取到,或者被中断,返回 false
  • unlock(),释放锁
  • newCondition(),返回此锁对象对应的条件实例,一般来说 Condition 和 Lock 是配合使用的

3. 成员变量

没有定义成员变量

4. 构造方法

默认情况下 ReentrantLock 是 非公平锁

  1. public ReentrantLock() {
  2. sync = new NonfairSync();
  3. }
  4. public ReentrantLock(boolean fair) {
  5. sync = fair ? new FairSync() : new NonfairSync();
  6. }

5. 成员方法

  1. // 具体的加锁由对应的实例去实现,实际调用的是 tryAquire 方法,默认情况是 非公平锁的 lock 方法
  2. public void lock() {
  3. sync.lock();
  4. }
  5. // 可中断的加锁办法,实际调用 AQS 中的 acquireInterruptibly 方法
  6. public void lockInterruptibly() throws InterruptedException {
  7. sync.acquireInterruptibly(1);
  8. }
  9. // 以非公平的方式获取锁
  10. public boolean tryLock() {
  11. return sync.nonfairTryAcquire(1);
  12. }
  13. // 释放锁
  14. public void unlock() {
  15. sync.release(1);
  16. }
  17. // 新建条件变量
  18. public Condition newCondition() {
  19. return sync.newCondition();
  20. }
  21. // 当前线程持有这个锁的数量
  22. public int getHoldCount() {
  23. return sync.getHoldCount();
  24. }
  25. // 当前线程是否持有这个锁
  26. public boolean isHeldByCurrentThread() {
  27. return sync.isHeldExclusively();
  28. }
  29. // 是否有线程持有当前锁
  30. public boolean isLocked() {
  31. return sync.isLocked();
  32. }
  33. // 是否是公平锁
  34. public final boolean isFair() {
  35. return sync instanceof FairSync;
  36. }
  37. // 获取当前锁对应的线程,可能为 null
  38. protected Thread getOwner() {
  39. return sync.getOwner();
  40. }
  41. // 查询是否有线程正在等待获取当前锁
  42. public final boolean hasQueuedThreads() {
  43. return sync.hasQueuedThreads();
  44. }
  45. // 查询给定的线程是否正在等待获取当前锁
  46. public final boolean hasQueuedThread(Thread thread) {
  47. return sync.isQueued(thread);
  48. }
  49. // 获取当前等待队列中的线程数,这个值随时变化
  50. public final int getQueueLength() {
  51. return sync.getQueueLength();
  52. }
  53. // 返回队列中正在等待锁的线程
  54. protected Collection<Thread> getQueuedThreads() {
  55. return sync.getQueuedThreads();
  56. }
  57. // 查询是否有线程 正在等待与此锁定有关的condition条件
  58. public boolean hasWaiters(Condition condition) {
  59. if (condition == null)
  60. throw new NullPointerException();
  61. if (!(condition instanceof AbstractQueuedSynchronizer.ConditionObject))
  62. throw new IllegalArgumentException("not owner");
  63. return sync.hasWaiters((AbstractQueuedSynchronizer.ConditionObject)condition);
  64. }
  65. // 返回与此锁有关的condition,由于condition条件不满足而进行等待的线程数
  66. public int getWaitQueueLength(Condition condition) {
  67. if (condition == null)
  68. throw new NullPointerException();
  69. if (!(condition instanceof AbstractQueuedSynchronizer.ConditionObject))
  70. throw new IllegalArgumentException("not owner");
  71. return sync.getWaitQueueLength((AbstractQueuedSynchronizer.ConditionObject)condition);
  72. }
  73. // 返回正在等待的线程
  74. protected Collection<Thread> getWaitingThreads(Condition condition) {
  75. if (condition == null)
  76. throw new NullPointerException();
  77. if (!(condition instanceof AbstractQueuedSynchronizer.ConditionObject))
  78. throw new IllegalArgumentException("not owner");
  79. return sync.getWaitingThreads((AbstractQueuedSynchronizer.ConditionObject)condition);
  80. }
  81. public String toString() {
  82. Thread o = sync.getOwner();
  83. return super.toString() + ((o == null) ?
  84. "[Unlocked]" :
  85. "[Locked by thread " + o.getName() + "]");
  86. }

内部类

ReentranLock 中的成员方法,基本是由继承于 AbstractQueuedSynchronizer 的内部类 Sync 的子类 FairSync 和 NonfairSync 实现的,所以弄懂 Sync 子类方法的实现,就基本弄懂了 ReentranLock 的整个源码。

  • Sync 是 ReentrantLock 的内部抽象类,继承于 AbstractQueuedSynchronizer
  • FairSync 和 NonfairSync 是 Sync 的子类,在父类的基础上,重写了 lock(),tryAquire() 方法,从而使得 ReentranLock 具有 XXX 特性。
  1. abstract static class Sync extends AbstractQueuedSynchronizer {
  2. private static final long serialVersionUID = -5179523762034025860L;
  3. // 抽象的 获取锁方法,供 Sync 的子类 FairSync 和 NonfairSync 使用
  4. abstract void lock();
  5. // 非公平方式获取锁
  6. final boolean nonfairTryAcquire(int acquires) {
  7. final Thread current = Thread.currentThread();
  8. int c = getState();
  9. if (c == 0) { // 若没有锁竞争
  10. if (compareAndSetState(0, acquires)) { // 若 state == 0,则以 CAS 方式将 state 设为 acquires
  11. setExclusiveOwnerThread(current); // 获取成功后,当前线程就具备这些资源的锁
  12. return true;
  13. }
  14. }
  15. else if (current == getExclusiveOwnerThread()) { // 当前线程已经拥有这个锁
  16. int nextc = c + acquires; // 重入的次数进行累加
  17. if (nextc < 0) // overflow
  18. throw new Error("Maximum lock count exceeded"
  19. setState(nextc); // 设定 state
  20. return true;
  21. }
  22. return false;
  23. }
  24. // 尝试释放锁,若当前线程释放之后不握有锁,返回 true,否则返回 false
  25. protected final boolean tryRelease(int releases) {
  26. int c = getState() - releases; // 释放 releases 个锁之后,还剩下 c 个锁
  27. if (Thread.currentThread() != getExclusiveOwnerThread()) // 如果当前线程不拥有这个锁,抛出异常
  28. throw new IllegalMonitorStateException();
  29. boolean free = false;
  30. if (c == 0) { // 释放 releases 个锁之后,当前线程握有 0 个锁,那么认为释放完成
  31. free = true;
  32. setExclusiveOwnerThread(null);
  33. }
  34. setState(c);
  35. return free;
  36. }
  37. // 当前线程是否是握有该锁的线程
  38. protected final boolean isHeldExclusively() {
  39. // While we must in general read state before owner,
  40. // we don't need to do so to check if current thread is owner
  41. return getExclusiveOwnerThread() == Thread.currentThread();
  42. }
  43. final ConditionObject newCondition() {
  44. return new ConditionObject();
  45. }
  46. // Methods relayed from outer class
  47. final Thread getOwner() {
  48. return getState() == 0 ? null : getExclusiveOwnerThread();
  49. }
  50. final int getHoldCount() {
  51. return isHeldExclusively() ? getState() : 0;
  52. }
  53. final boolean isLocked() {
  54. return getState() != 0;
  55. }
  56. /**
  57. * Reconstitutes the instance from a stream (that is, deserializes it).
  58. */
  59. private void readObject(java.io.ObjectInputStream s)
  60. throws java.io.IOException, ClassNotFoundException {
  61. s.defaultReadObject();
  62. setState(0); // reset to unlocked state
  63. }
  64. }

FairSync

实现父类未实现的 lock 方法,实际调用的是 AQS 的 acquire 方法,而 AQS 的 acquire 在 FairSync 中又进行了重写实现。

相比非公平锁,公平锁的 tryAcquire 的实现就多了一个 hasQueuedPredecessors

  1. static final class FairSync extends Sync {
  2. private static final long serialVersionUID = -3000897897090466540L;
  3. final void lock() {
  4. acquire(1);
  5. }
  6. /**
  7. * Fair version of tryAcquire. Don't grant access unless
  8. * recursive call or no waiters or is first.
  9. */
  10. protected final boolean tryAcquire(int acquires) {
  11. final Thread current = Thread.currentThread();
  12. int c = getState();
  13. if (c == 0) {
  14. // 阻塞队列中没有比当前线程等待更久的线程 并且 state 的值是 0 时,以 CAS 方式获取独占锁并返回 true
  15. if (!hasQueuedPredecessors() &&
  16. compareAndSetState(0, acquires)) {
  17. setExclusiveOwnerThread(current);
  18. return true;
  19. }
  20. }
  21. else if (current == getExclusiveOwnerThread()) { // 当前线程已经获取了该锁,属于重入的情况,那么直接进行累加
  22. int nextc = c + acquires;
  23. if (nextc < 0)
  24. throw new Error("Maximum lock count exceeded");
  25. setState(nextc);
  26. return true;
  27. }
  28. return false;
  29. }
  30. }

NonfairSync

  1. static final class NonfairSync extends Sync {
  2. private static final long serialVersionUID = 7316153563782823691L;
  3. /**
  4. * Performs lock. Try immediate barge, backing up to normal
  5. * acquire on failure.
  6. */
  7. final void lock() {
  8. if (compareAndSetState(0, 1)) // 上来直接判断 state,若没有竞争,直接加锁
  9. setExclusiveOwnerThread(Thread.currentThread());
  10. else
  11. acquire(1); // 否则和 Fair.lock 一样调用 AQS 的 aquire 方法
  12. }
  13. protected final boolean tryAcquire(int acquires) {
  14. return nonfairTryAcquire(acquires); // 以非公平形式获取锁
  15. }
  16. }

AQS 的 acquire 方法,这个方法大体意思:

  • 直接以 CAS 方式先尝试获取锁,获取成功就直接返回
  • 获取失败,那么添加到队列末尾,添加成功,直接返回
  • 若队列末尾添加失败,该线程就进行中断
  1. AbstractQueuedSynchronizer.java
  2. public final void acquire(int arg) {
  3. if (!tryAcquire(arg) &&
  4. acquireQueued(addWaiter(Node.EXCLUSIVE), arg))
  5. selfInterrupt();
  6. }
  7. static void selfInterrupt() {
  8. Thread.currentThread().interrupt();
  9. }
  10. // 当队列为空 或者 队列中有比当前线程等待更久的线程时,返回 true
  11. public final boolean hasQueuedPredecessors() {
  12. // The correctness of this depends on head being initialized
  13. // before tail and on head.next being accurate if the current
  14. // thread is first in queue.
  15. Node t = tail; // Read fields in reverse initialization order
  16. Node h = head;
  17. Node s;
  18. return h != t &&
  19. ((s = h.next) == null || s.thread != Thread.currentThread());
  20. }

AQS 中的方法

  1. public final void acquireInterruptibly(int arg)
  2. throws InterruptedException {
  3. if (Thread.interrupted())
  4. throw new InterruptedException();
  5. if (!tryAcquire(arg)) // tryAcquire 由子类去实现,去获取 arg 个锁,获取失败就抛出异常
  6. doAcquireInterruptibly(arg);
  7. }

ReentrantLock 与 Synchronized 区别

  1. ReentrantLock 显示的获得、释放锁,synchronized 隐式获得释放锁

  2. ReentrantLock 可响应中断、可轮回,synchronized 是不可以响应中断的,为处理锁的不可用性提供了更高的灵活性

  3. ReentrantLock 是 API 级别的,synchronized 是 JVM 级别的

  4. ReentrantLock 可以实现公平锁

  5. ReentrantLock 通过 Condition 可以绑定多个条件

  6. 底层实现不一样, synchronized 是同步阻塞,使用的是悲观并发策略,lock 是同步非阻塞,采用的是乐观并发策略

  7. Lock 是一个接口,而 synchronized 是 Java 中的关键字,synchronized 是内置的语言实现。

  8. synchronized 在发生异常时,会自动释放线程占有的锁,因此不会导致死锁现象发生;而 Lock 在发生异常时,如果没有主动通过 unLock() 去释放锁,则很可能造成死锁现象,因此使用 Lock 时需要在 finally 块中释放锁。

  9. Lock 可以让等待锁的线程响应中断,而 synchronized 却不行,使用 synchronized 时,等待的线程会一直等待下去,不能够响应中断。

  10. 通过 Lock 可以知道有没有成功获取锁,而 synchronized 却无法办到。

  11. Lock 可以提高多个线程进行读操作的效率,既就是实现读写锁等。

7. 总结

  • ReentrantLock 的 tryLock() 方法是非公平的,因为无论在什么模式下,该方法都会调用抽象同步器的 nonfairTryAcquire() 方法