1 示例

  1. public class ReentrantLockTest {
  2. public static void main(String[] args) {
  3. ThreadDemo demo = new ThreadDemo();
  4. ThreadDemo2 lockDemo = new ThreadDemo2();
  5. Thread t = new Thread(demo);
  6. Thread t2 = new Thread(demo);
  7. Thread tt = new Thread(lockDemo);
  8. Thread tt2 = new Thread(lockDemo);
  9. t.start();
  10. t2.start();
  11. tt.start();
  12. tt2.start();
  13. }
  14. }
  15. class ThreadDemo implements Runnable {
  16. volatile int num = 0;
  17. ReentrantLock lock = new ReentrantLock();
  18. @Override
  19. public void run() {
  20. lock.lock();
  21. for (int i = 0; i < 1000000; i++) {
  22. num++;
  23. }
  24. System.out.println("加锁操作后,num:" + num);
  25. lock.unlock();
  26. }
  27. }
  28. class ThreadDemo2 implements Runnable {
  29. int num = 0;
  30. @Override
  31. public void run() {
  32. for (int i = 0; i < 1000000; i++) {
  33. num++;
  34. }
  35. System.out.println("不加锁,num:" + num);
  36. }
  37. }

输出:

  1. 不加锁,num:1123472
  2. 加锁操作后,num:1000000
  3. 不加锁,num:1979071
  4. 加锁操作后,num:2000000

2 类结构关系

ReentrantLock 是可重入的独占锁 只能有 线程可 获取该锁,其 获取该锁的线程会被阻塞而被放入该锁的 阻塞队列里面。
image.png
ReentrantLock 最终还是使用 AQS 来实 的,并且根据参数来决定其内部是一个公平还是非公平锁,默认是非公平锁。

  1. //无参构造,默认是非公平锁
  2. public ReentrantLock() {
  3. sync = new NonfairSync();
  4. }
  5. //根据参数来决定其内部是一个公平还是非公平锁
  6. public ReentrantLock(boolean fair) {
  7. sync = fair ? new FairSync() : new NonfairSync();
  8. }

其中 Sync 类直接继承自 AQS 它的子类 No fairSy FairSync 分别实现了获取锁的非公平与公平策略 。
在这里, AQS state 态值表示线程获取该锁的可重入次数 在默认情况下, state的值为 表示当前锁没有被任何线程持有 线程第 次获取该锁 会尝试使用 CAS 设置 state 的值为 ,如果 CAS 成功则当前线程获取了该锁,然后记录该锁的持有者为当前线程 在该线程没有释放锁的情况下第 次获取该锁后 ,状态值被设置为 这就是可重入次数 在该线程释放该锁时,会尝试使用 CAS 让状态值减 如果减 后状态值为 0,则当前线程释放该锁。

3 NonfairSync实现原理

3.1 lock

  1. public void lock() {
  2. sync.lock();
  3. }
  1. final void lock() {
  2. if (compareAndSetState(0, 1))
  3. //CAS成功,设置该锁持有者是当前线程
  4. setExclusiveOwnerThread(Thread.currentThread());
  5. else
  6. //CAS失败,尝试获取锁
  7. acquire(1);
  8. }

3.2 acquire

  1. public final void acquire(int arg) {
  2. if (!tryAcquire(arg) &&
  3. //获取锁失败,会把当前线程放入AQS 阻塞队列
  4. acquireQueued(addWaiter(Node.EXCLUSIVE), arg))
  5. selfInterrupt();
  6. }

3.3 tryAcquire

  1. protected final boolean tryAcquire(int acquires) {
  2. return nonfairTryAcquire(acquires);
  3. }
  1. final boolean nonfairTryAcquire(int acquires) {
  2. final Thread current = Thread.currentThread();
  3. int c = getState();
  4. if (c == 0) { //当前AQS state状态值是0(无锁);
  5. if (compareAndSetState(0, acquires)) {
  6. setExclusiveOwnerThread(current); //CAS成功,设置该锁持有者是当前线程
  7. return true;
  8. }
  9. }
  10. else if (current == getExclusiveOwnerThread()) { //当前线程是该锁持有者
  11. int nextc = c + acquires; //状态值加1
  12. if (nextc < 0) // overflow
  13. throw new Error("Maximum lock count exceeded");
  14. setState(nextc);
  15. return true;
  16. }
  17. return false; //都不是,返回false,丢入队列
  18. }

3.4 unlock

  1. public void unlock() {
  2. sync.release(1);
  3. }
  1. public final boolean release(int arg) {
  2. if (tryRelease(arg)) {
  3. Node h = head;
  4. if (h != null && h.waitStatus != 0)
  5. unparkSuccessor(h);
  6. return true;
  7. }
  8. return false;
  9. }

3.5 tryRelease

此方法为抽象类Sync的final方法,公平锁和非公平锁公用此方法

  1. protected final boolean tryRelease(int releases) {
  2. int c = getState() - releases; //重入次数减1
  3. //如果当前线程不是锁持有者,抛异常
  4. if (Thread.currentThread() != getExclusiveOwnerThread())
  5. throw new IllegalMonitorStateException();
  6. boolean free = false;
  7. if (c == 0) { //如果当前可重入次数为0,清空锁
  8. free = true;
  9. setExclusiveOwnerThread(null);
  10. }
  11. setState(c); //重入次数减1
  12. return free;
  13. }

4 FairSync 实现原理

公平锁的话只需要看 FairSync 重写 tryAcquire方法。

4.1 tryAcquire

  1. protected final boolean tryAcquire(int acquires) {
  2. final Thread current = Thread.currentThread();
  3. int c = getState();
  4. if (c == 0) { //当前AQS state状态值是0(无锁);
  5. // 只有队列为空时,才会通过CAS尝试获取锁
  6. if (!hasQueuedPredecessors() &&
  7. compareAndSetState(0, acquires)) {
  8. setExclusiveOwnerThread(current);
  9. return true;
  10. }
  11. }
  12. else if (current == getExclusiveOwnerThread()) { //当前线程是该锁持有者
  13. int nextc = c + acquires;
  14. if (nextc < 0)
  15. throw new Error("Maximum lock count exceeded");
  16. setState(nextc);
  17. return true;
  18. }
  19. return false;
  20. }
  21. }

4.2 hasQueuedPredecessors

  1. public final boolean hasQueuedPredecessors() {
  2. // The correctness of this depends on head being initialized
  3. // before tail and on head.next being accurate if the current
  4. // thread is first in queue.
  5. Node t = tail; // Read fields in reverse initialization order
  6. Node h = head;
  7. Node s;
  8. //如果 h==t 说明当前队列为空,直接返回 false
  9. return h != t &&
  10. ((s = h.next) == null || s.thread != Thread.currentThread());
  11. }

在如上代码中,如果当前线程节点有前驱节点则返回住时, 如果当前 AQ 队列为空或者当前线程节点是 AQS 第一个节点则返回 false 。其中如果 h==t 说明当前队列为空,直接返回 false ;如果 h!=t 并且 ==null 则说明有一个元素将要作为 QS 第一个节点入队列 (回顾前面的内容, enq 数的第一个元素入队列是两步操作: 首先 建一个哨兵头节点,然后将第一 元素插入哨兵节点后 ,那么返回 true ,如果 h!=t 并且 s!=null和s.thr巳ad != Thread.cunentThread()则说明队列 面的第 个元 不是当前线程,那么返回true