ReentranLock示例
我们都知道ReentranLock
的加锁行为和Synchronized
类似,都是可重入的锁,但是二者的实现方式确实完全不同的,我们之后也会讲解Synchronized
的原理.除此之外,Synchronized的阻塞无法被中断,而ReentrantLock则提供了可中断的阻塞 下面的代码是ReentranLock
的相关API,我们就以此为顺序,依次讲解.
1 2 3 ReentrantLock lock = new ReentrantLock(); lock.lock(); lock.unlock();
公平锁和非公平锁
ReentranLock分为公平锁和非公平锁.二者的区别就在获取锁是否和排队顺序相关.我们都知道,如果当前锁被另一个线程持有,那么当前申请锁的线程会被挂起等待,然后加入一个等待队列里.理论上,先调用lock
函数被挂起等待的线程应该排在等待队列的前端,后调用的就排在后边.如果此时,锁被释放,需要通知等待线程再次尝试获取锁,公平锁会让最先进入队列的线程获得锁,而非公平锁则会唤醒所有线程,让它们再次尝试获取锁,所以可能会导致后来的线程先获得了锁,则就是非公平.
1 2 3 public ReentrantLock (boolean fair) { sync = fair ? new FairSync() : new NonfairSync(); }
我们会发现FairSync
和NonfairSync
都继承了Sync
类,而Sync
的父类就是AbstractQueuedSynchronizer
.但是AQS
的构造函数是空的,并没有任何操作. 之后的源码分析,如果没有特别说明,就是指公平锁.
lock操作
ReentranLock
的lock
函数如下所示,直接调用了sync
的lock
函数.也就是调用了FairSync
的lock
函数.
1 2 3 4 5 6 7 8 //ReentranLock public void lock() { sync.lock(); } //FairSync final void lock() { acquire(1);//调用了AQS的acquire函数,这是关键函数之一 }
好,我们接下来就正式开始AQS
相关的源码分析了,acquire
函数你可以将其理解为获取一个同一时间只能有一个函数获取的量,这个量就是锁概念的抽象化.我们先分析代码,你慢慢就会明白其中的含义.
1 2 3 4 5 6 7 public final void acquire (int arg) { if (!tryAcquire(arg) && acquireQueued(addWaiter(Node.EXCLUSIVE), arg)) selfInterrupt(); }
tryAcquire
,addWaiter
和acquireQueued
都是十分重要的函数,我们接下来依次学习一下这些函数,理解它们的作用.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 //AQS类中的变量. private volatile int state; //这是FairSync的实现,AQS中未实现,子类按照自己的需要实现该类 protected final boolean tryAcquire(int acquires) { final Thread current = Thread.currentThread(); //获取AQS中的state变量,代表抽象概念的锁. int c = getState(); if (c == 0) { //值为0,那么当前独占性变量还未被线程占有 if (!hasQueuedPredecessors() && //如果当前阻塞队列上没有先来的线程在等待,UnfairSync这里的实现就不一致 compareAndSetState(0, acquires)) { //成功cas,那么代表当前线程获得该变量的所有权,也就是说成功获得锁 setExclusiveOwnerThread(current); return true; } } else if (current == getExclusiveOwnerThread()) { //如果该线程已经获取了独占性变量的所有权,那么根据重入性 //原理,将state值进行加1,表示多次lock //由于已经获得锁,该段代码只会被一个线程同时执行,所以不需要 //进行任何并行处理 int nextc = c + acquires; if (nextc < 0) throw new Error("Maximum lock count exceeded"); setState(nextc); return true; } //上述情况都不符合,说明获取锁失败 return false; }
由上述代码我们可以发现,tryAcquire
就是尝试获取那个线程独占的变量state
.state的值表示其状态:如果是0,那么当前还没有线程独占此变量;否在就是已经有线程独占了这个变量,也就是代表已经有线程获得了锁.但是这个时候要再进行一次判断,看是否是当前线程自己获得的这个锁,如果是,那么就增加state的值.
ReentranLock获得锁
这里有几点需要说明一下,首先是compareAndSetState
函数,这是使用CAS操作来设置state
的值,而且state值设置了volatile
修饰符,通过这两点来确保修改state的值不会出现多线程问题.然后是公平锁和非公平锁的区别问题,在UnfairSync
的nonfairTryAcquire
函数中不会在相同的位置上调用hasQueuedPredecessors
来判断当前是否已经有线程在排队等待获得锁. 如果tryAcquire
返回true
,那么就是获取锁成功,如果返回false,那么就是未获得锁.需要加入阻塞等待队列.我们下面就来看一下addWaiter
的相关操作
等待锁的阻塞队列
将保存当前线程信息的节点加入到等待队列的相关函数中涉及到了无锁队列的相关算法,由于在AQS
中只是将节点添加到队尾,使用到的无锁算法也相对简单.真正的无锁队列的算法我们等到分析ConcurrentSkippedListMap
时在进行讲解.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 private Node addWaiter(Node mode) { Node node = new Node(Thread.currentThread(), mode); //先使用快速如列法来尝试一下,如果失败,则进行更加完备的如列算法. Node pred = tail;//列尾指针 if (pred != null) { node.prev = pred; //步骤1:该节点的前趋指针指向tail if (compareAndSetTail(pred, node)){ //步骤二:cas将尾指针指向该节点 pred.next = node;//步骤三:如果成果,让旧列尾节点的next指针指向该节点. return node; } } //cas失败,或在pred == null时调用enq enq(node); return node; } private Node enq(final Node node) { for (;;) { //cas无锁算法的标准for循环,不停的尝试 Node t = tail; if (t == null) { //初始化 if (compareAndSetHead(new Node())) //需要注意的是head是一个哨兵的作用,并不代表某个要获取锁的线程节点 tail = head; } else { //和addWaiter中一致,不过有了外侧的无限循环,不停的尝试,自旋锁 node.prev = t; if (compareAndSetTail(t, node)) { t.next = node; return t; } } } }
通过调用addWaiter
函数,AQS
将当前线程加入到了等待队列,但是还没有阻塞当前线程的执行,接下来我们就来分析一下acquireQueued
函数.
等待队列节点的操作
由于进入阻塞状态的操作会降低执行效率,所以,AQS
会尽力避免试图获取独占性变量的线程进入阻塞状态.所以,当线程加入等待队列之后,acquireQueued
会执行一个for循环,每次都判断当前节点是否应该获得这个变量(在队首了),如果不应该获取或在再次尝试获取失败,那么就调用shouldParkAfterFailedAcquire
判断是否应该进入阻塞状态,如果当前节点之前的节点已经进入阻塞状态了,那么就可以判定当前节点不可能获取到锁,为了防止CPU不停的执行for循环,消耗CPU资源,调用parkAndCheckInterrupt
函数来进入阻塞状态.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 final boolean acquireQueued (final Node node, int arg) { boolean failed = true ; try { boolean interrupted = false ; for (;;) { final Node p = node.predecessor(); if (p == head && tryAcquire(arg)) { setHead(node); p.next = null ; failed = false ; return interrupted; } if (shouldParkAfterFailedAcquire(p, node) && parkAndCheckInterrupt()) interrupted = true ; } } finally { if (failed) cancelAcquire(node); } } private static boolean shouldParkAfterFailedAcquire (Node pred, Node node) { int ws = pred.waitStatus; if (ws == Node.SIGNAL) return true ; if (ws > 0 ) { do { node.prev = pred = pred.prev; } while (pred.waitStatus > 0 ); pred.next = node; } else { compareAndSetWaitStatus(pred, ws, Node.SIGNAL); } return false ; } private final boolean parkAndCheckInterrupt () { LockSupport.park(this ); return Thread.interrupted(); }
阻塞和中断
由上述分析,我们知道了AQS
通过调用LockSupport
的park
方法来执行阻塞当前进程的操作.其实,这里的阻塞就是线程不再执行的含义.通过调用这个函数,线程进入阻塞状态,上述的lock
操作也就阻塞了.等待中断或在独占性变量被释放.
1 2 3 4 5 6 public static void park (Object blocker) { Thread t = Thread.currentThread(); setBlocker(t, blocker); UNSAFE.park(false , 0L ); setBlocker(t, null ); }
关于中断的相关知识,我们以后再说,就继续沿着AQS
的主线,看一下释放独占性变量的相关操作吧.
ReentrantLock未获得阻塞,加入队列
unlock操作
与lock
操作类似,unlock
操作调用了AQS
的relase
方法,参数和调用acquire
时一样,都是1.
1 2 3 4 5 6 7 8 9 public final boolean release(int arg) { if (tryRelease(arg)) { //释放独占性变量,起始就是将status的值减1,因为acquire时是加1 Node h = head; if (h != null && h.waitStatus != 0) unparkSuccessor(h);//唤醒head的后继节点 return true; } return false; }
由上述代码可知,release就是先调用tryRelease
来释放独占性变量,如果成果,那么就看一下是否有等待锁的阻塞线程,如果有,就调用unparkSuccessor
来唤醒他们.
1 2 3 4 5 6 7 8 9 10 11 12 13 protected final boolean tryRelease(int releases) { //由于只有一个线程可以获得独占先变量,所以,所有操作不需要考虑多线程 int c = getState() - releases; if (Thread.currentThread() != getExclusiveOwnerThread()) throw new IllegalMonitorStateException(); boolean free = false; if (c == 0) { //如果等于0,那么说明锁应该被释放了,否在表示当前线程有多次lock操作. free = true; setExclusiveOwnerThread(null); } setState(c); return free; }
我们可以看到tryRelease
中的逻辑也体现了可重入锁的概念,只有等到state
的值为1时,才代表锁真正被释放了.所以独占性变量state
的值就代表锁的有无.当state=0
时,表示锁未被占有,否在表示当前锁已经被占有.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 private void unparkSuccessor (Node node) { ..... Node s = node.next; if (s == null || s.waitStatus > 0 ) { s = null ; for (Node t = tail; t != null && t != node; t = t.prev) if (t.waitStatus <= 0 ) s = t; } if (s != null ) LockSupport.unpark(s.thread); }
调用了unpark
方法后,进行lock
操作被阻塞的线程就恢复到运行状态,就会再次执行acquireQueued
中的无限for循环中的操作,再次尝试获取锁.
ReentrantLock释放锁并通知阻塞线程恢复执行
后记
有关AQS
和ReentrantLock
的分析就差不多结束了,不得不说,我第一次看到AQS的实现时真是震惊,以前都认为Synchronized
和ReentrantLock
的实现原理是一致的,都是依靠java虚拟机的功能实现的,没有想到还有AQS
这样一个背后大Boss在提供帮助啊.学习了这个类的原理,我们对JUC的很多类的分析就简单了很多,此外,AQS
涉及的CAS
操作和无锁队列的算法也为我们学习其他无锁算法提供了基础.知识的海洋是无限的啊!
转自:http://remcarpediem.com/2017/04/06/AbstractQueuedSynchronizer%E8%B6%85%E8%AF%A6%E7%BB%86%E5%8E%9F%E7%90%86%E8%A7%A3%E6%9E%90/