AQS之ReentrantLock源码解析

概述

在编码中常使用ReentrantLock时候,它可以实现线程在获取锁时候公平与非公平。所谓公平在排队者挨个获取锁,非公平排队者第一个可能和插队者争抢锁。我们想来上一个类图了解他们之间关系。


关系图

源码分析

我们先从ReentrantLock构造方法开始分析。

public ReentrantLock() {
        sync = new NonfairSync();
    }
 public ReentrantLock(boolean fair) {
        sync = fair ? new FairSync() : new NonfairSync();
    }

从代码可以看出,默认采用非公平锁。接下来了解,公平锁和非公平锁他们代码具体差异在哪里。

    //公平锁
    static final class FairSync extends Sync {
 
        final void lock() {
            //1参数为1 表示要去获得锁
            acquire(1);
        }
  
    //非公平锁
    static final class NonfairSync extends Sync {
        final void lock() {
            //2 通过CAS直接加锁,如果锁未被他人使用,
            if (compareAndSetState(0, 1))
                //3设置当前执行的线程
                setExclusiveOwnerThread(Thread.currentThread());
            else
                acquire(1);
        }
    }
  • 注释1,公平锁,直接独占方式获取锁。
  • 注释2,非公平锁,通过CAS直接加锁,如果锁未被他人使用,修改成功获得锁。如果没有获取到锁然后调用 acquire(1)方法。
  • 注释3,设置当前执行的线程。
    我们继续看上面类是如何尝试获取锁的。
    //公平锁
    static final class FairSync extends Sync {

         protected final boolean tryAcquire(int acquires) {
            final Thread current = Thread.currentThread();
            int c = getState();
            if (c == 0) {
                //1  此方法保证获得锁根据队列来 先来先获得锁
                if (!hasQueuedPredecessors() &&
                    compareAndSetState(0, acquires)) {
                    setExclusiveOwnerThread(current);
                    return true;
                }
            }
            // 可重入锁
            else if (current == getExclusiveOwnerThread()) {
                int nextc = c + acquires;
                if (nextc < 0)
                    throw new Error("Maximum lock count exceeded");
                setState(nextc);
                return true;
            }
            return false;
        }
    }


    //非公平锁
    static final class NonfairSync extends Sync {
      //2 尝试在去获得锁
        protected final boolean tryAcquire(int acquires) {
            return nonfairTryAcquire(acquires);
        }
    }

   abstract static class Sync extends AbstractQueuedSynchronizer {
         // 尝试获取锁
        final boolean nonfairTryAcquire(int acquires) {
            final Thread current = Thread.currentThread();
            int c = getState();
            if (c == 0) {
                //3 尝试获取锁,如果锁未被使用。先进入的线程
                if (compareAndSetState(0, acquires)) {
                    setExclusiveOwnerThread(current);
                    return true;
                }
            }
            //4 如果锁被自己占用,自己获取锁可重入锁
            else if (current == getExclusiveOwnerThread()) {
                int nextc = c + acquires;
                if (nextc < 0) // overflow
                    throw new Error("Maximum lock count exceeded");
                setState(nextc);
                return true;
            }
            return false;
        }
  }
//5 判断头结点是否是当前线程
public abstract class AbstractQueuedSynchronizer
    extends AbstractOwnableSynchronizer{
    public final boolean hasQueuedPredecessors() {
        Node t = tail; // Read fields in reverse initialization order
        Node h = head;
        Node s;
        return h != t &&
            ((s = h.next) == null || s.thread != Thread.currentThread());
    }
}

  • 注释1,公平锁,hasQueuedPredecessors方法判断头结点(AQS后面讲,线程转换成双向链表才存储)是否是当前线程。如果是(按照链表顺序执行)当前线程,直接尝试获取锁。
  • 注释2,非公平锁,尝试获取锁
  • 注释3,发起线程尝试获取锁,如果锁未被使用则获取到搜,这里和链表的head存在竞争关系,锁被争抢。
  • 注释4,如果锁被自己占用,自己获取锁可重入锁。
  • 注释5,判断头结点是否是当前线程。

接下来我们来分析AQS(AbstractQueuedSynchronizer)源码,不管是公平锁还是非公平锁,以下实现机制都一样。

public final void acquire(int arg) {
        if (!tryAcquire(arg) &&
            acquireQueued(addWaiter(Node.EXCLUSIVE), arg))
            selfInterrupt();
    }

尝试获取锁,如果没有获取到锁,把当前线程封装成Node节点,存入链表中。

 static final class Node {
        /** Marker to indicate a node is waiting in shared mode */
        static final Node SHARED = new Node();
        /** Marker to indicate a node is waiting in exclusive mode */
        static final Node EXCLUSIVE = null;
        /** waitStatus value to indicate thread has cancelled */
        static final int CANCELLED =  1;
        /** waitStatus value to indicate successor's thread needs unparking */
        static final int SIGNAL    = -1;
        /** waitStatus value to indicate thread is waiting on condition */
        static final int CONDITION = -2;
        static final int PROPAGATE = -3;
        volatile int waitStatus;
        volatile Node prev;
        volatile Node next;
        volatile Thread thread;
        Node nextWaiter;
        
        final Node predecessor() throws NullPointerException {
            Node p = prev;
            if (p == null)
                throw new NullPointerException();
            else
                return p;
        }

把线程封装成Node,且有自己的状态waitStatus。

//1 为获取锁的线程,添加等待节点
    private Node addWaiter(Node mode) {
        Node node = new Node(Thread.currentThread(), mode);
        // Try the fast path of enq; backup to full enq on failure
        // 2 tail 尾结点
        Node pred = tail;
        // 尾结点不为空
        if (pred != null) {
            //3 在尾结点添加新节点
            node.prev = pred;
            //4 修改尾结点的内存地址内存,tail = node
            if (compareAndSetTail(pred, node)) {
                pred.next = node;
                return node;
            }
        }
        enq(node);
        return node;
    }
  // 创建tail节点内容,第一个节点为哨兵节点,第二个节点为node
    private Node enq(final Node node) {
        for (;;) {
            Node t = tail;
            //5 尾结点为空,创建哨兵节点(Node都是一个空对象)
            if (t == null) { // Must initialize
                //6 tail head 都为哨兵节点
                if (compareAndSetHead(new Node()))
                    tail = head;
            } else {
                //7 为node 指定 prev节点,首先初始化哨兵节点,然后把哨兵节点作为node的prev。
                node.prev = t;
                //8 尾结点变化成 node
                if (compareAndSetTail(t, node)) {
                    t.next = node;
                    return t;
                }
            }
        }
    }
  • 注释1,为获取锁的线程,添加等待节点。
  • 注释2,拿到尾结点。
  • 注释3,如果pre节点不为空,添加新节点到尾结点处。
  • 注释4,修改尾结点的内存地址内存的值,tail = node。
  • 注释5,尾结点为空,创建哨兵节点(Node都是一个空对象)。
  • 注释6,给tail和head 赋值哨兵节点。
  • 注释7,为node 指定 prev节点,首先初始化哨兵节点,然后把哨兵节点作为node的prev。
  • 注释8,尾结点变化成 node。
    总结,上面两个方法主要做的事情是,创建哨兵节点,下一个节点为node。
 final boolean acquireQueued(final Node node, int arg) {
        boolean failed = true;
        try {
            boolean interrupted = false;
            for (;;) {
                //1 获得node的上一个节点
                final Node p = node.predecessor();
                //2 p节点是头结点,通过调用tryAcquire方法获得锁
                if (p == head && tryAcquire(arg)) {
                    //3 把node节点变成哨兵节点
                    setHead(node);
                    p.next = null; // help GC
                    failed = false;
                    return interrupted;
                }
                //4 Node状态变成等待,删除线程取消的节点。parkAndCheckInterrupt等待获取锁
                if (shouldParkAfterFailedAcquire(p, node) &&
                    parkAndCheckInterrupt())
                    interrupted = true;
            }
        } finally {
            if (failed)
                cancelAcquire(node);
        }
    }
// 进入阻塞状态
private final boolean parkAndCheckInterrupt() {
        LockSupport.park(this);
        return Thread.interrupted();
    }
private static boolean shouldParkAfterFailedAcquire(Node pred, Node node) {
        int ws = pred.waitStatus;
        //5 等待获取锁状态
        if (ws == Node.SIGNAL)
            return true;
        // 6 线程被取消
        if (ws > 0) {
            do {
                //7 删除被取消的线程节点
                node.prev = pred = pred.prev;
            } while (pred.waitStatus > 0);
            pred.next = node;
        } else {
            compareAndSetWaitStatus(pred, ws, Node.SIGNAL);
        }
        return false;
    }
  • 注释1,获得node(新线程)的上一个节点。
  • 注释2,p节点是头结点,通过调用tryAcquire方法获得锁。
  • 注释3,把node节点变成哨兵节点。
  • 注释4,Node状态变成等待,删除线程取消的节点。parkAndCheckInterrupt等待获取锁。
  • 注释5,等待获取锁状态。
  • 注释6,被标识为取消的节点。
  • 注释7,删除被取消的线程节点。
    我们了解到通过AQS给需要加锁的线程,封装成Node双向链表。接下来我们看看如何唤醒阻塞的线程。
    //ReentrantLock.java
    // 释放锁
    public void unlock() {
        sync.release(1);
    }

//AbstractQueuedSynchronizer.java
public final boolean release(int arg) {
        if (tryRelease(arg)) {
          //1 获取得到头结点
            Node h = head;
            if (h != null && h.waitStatus != 0)
                unparkSuccessor(h);
            return true;
        }
        return false;
    }
 private void unparkSuccessor(Node node) {
        int ws = node.waitStatus;
        if (ws < 0)
            compareAndSetWaitStatus(node, ws, 0);
        //2 拿到头结点下一个节点,都是等待获取锁的节点
        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;
        }
        // 给节点s,发一张允许票
        if (s != null)
            LockSupport.unpark(s.thread);
    }
  • 注释1,获取得到头结点。
  • 注释2,拿到头结点下一个节点,都是等待获取锁的节点。
    以上是所有线程使用ReentrantLock原理,如有描述错误地方请大家多多指正。
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,732评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,496评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,264评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,807评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,806评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,675评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,029评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,683评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,704评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,666评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,773评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,413评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,016评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,978评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,204评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,083评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,503评论 2 343

推荐阅读更多精彩内容