AQS框架

AQS(AbstractQueuedSynchronized)队列同步器
理解AQS需要知道1.队列结构,2.同步的对象
皮皮甜觉得它是并发的基础,因为比如我们常用的ReentrantLock、CountDownLatch底层实现都是基于AQS框架的,所以深入理解AQS非常有必要
废话不多说,进入主题。
在分析源码之前,先说说AQS使用的设计模式:模板方法模式,什么是模板方法模式可以参考之前写的设计模式---模板方法模式
AQS中可重写的方法:
1.protected boolean tryAcquire(int arg)
2.protected boolean tryRelease(int arg)
3.protected boolean tryAcquireShared(int arg)
4.protected boolean tryReleaseShared(int arg)
5.protected boolean isHeldExclusively()
其中1、2为独占式获得锁和释放锁
3、4为共享式获得锁和释放锁
5判断锁是否被当前线程所占用
独占式和共享式的区别:是否允许多个线程获取锁

AQS中的模板方法:
1.void acquire(int arg)
2.void acquireInterruptibly(int arg)
3.boolean tryAcquireNanos(int arg, long nanos)
4.void acquireShared(int arg)
5.boolean releaseShared(int arg)
等等,模板方法有很多,这里只举例一些,看到3、4方法就知道为什么ReentrantLock支持超时获取锁、以及可中断了

因为是基于模板方法的,所以我们自己使用AQS框架时,会重写可重写的方法,模板方法中调用的是可重写的方法。其中在可重写方法中,我们通过调用setState(int newState)getState()compareAndSetState(int expect, int update)方法对同步状态进行修改

还是举个例子来说明AQS框架的使用:
TwinsLock.java

public class TwinsLock implements Lock {
    //自定义同步器
    private static class Sync extends AbstractQueuedLongSynchronizer {
        //重写AQS中的一些方法

        public Sync(int count) {
            if (count <= 0) {
                throw new IllegalArgumentException("count must large than zero.");
            }
            setState(count);  //设置同步状态,表示同时允许多少线程访问共享资源
        }

        //重写共享式获取锁和释放锁的方法
        @Override
        protected long tryAcquireShared(long reduceCount) {
            for (;;) {
                int current = (int)getState();  //获得此时的同步状态
                int newCount = (int) (current - reduceCount);

                //获取同步状态失败
                if (newCount < 0 || compareAndSetState(current, newCount)) {
                    return newCount;
                }
            }
        }

        @Override
        protected boolean tryReleaseShared(long returnCount) {
            for (;;) {
                int current = (int)getState();
                int newCount = (int)(current + returnCount);
                if (compareAndSetState(current, newCount));  //重新设置共享状态
            }
        }
    }
    private final Sync sync = new Sync(2);  //设置同步状态为2
    //Lock中的接口
    @Override
    public void lock() {
        sync.acquireShared(1);
    }

    @Override
    public void lockInterruptibly() throws InterruptedException {

    }

    @Override
    public boolean tryLock() {
        return false;
    }

    @Override
    public boolean tryLock(long time, TimeUnit unit) throws InterruptedException {
        return false;
    }

    @Override
    public void unlock() {
        sync.releaseShared(1);
    }

    @Override
    public Condition newCondition() {
        return null;
    }
}

这里定义了一种锁,允许有两个线程获得锁,因为允许两个线程,所以使用共享式模式。
一般会将继承AQS框架的类作为内部类,即这里的Sync.java
独占式模式核心方法源码分析:
重写了tryAcquireShared(long reduceCount)tryReleaseShared(long returnCount)这两个方法,在TwinsLock.java中,我们通过sync调用模板方法sync.acquireShared(1)sync.releaseShared(1)

1.独占式获得锁

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

可以看到,在模板方法acquire(int arg)里调用了可重写方法tryAcquire(arg)
1)成功获得锁,即tryAcquire(arg)返回true
2)获得锁失败,就要将当前线程和同步状态构造成一个节点加入队列中,同时阻塞当前线程并进行自旋判断是否可以获得锁,这几步主要是通过addWaiter(Node mode)acquireQueued(final Node node, int arg)方法实现的
addWaiter(Node mode)方法往队列中添加节点

private Node addWaiter(Node mode) {
        //1.将当前线程和同步状态构造成为一个节点Node
        Node node = new Node(Thread.currentThread(), mode);
        //2.获得队列的尾节点 
        Node pred = tail;
        if (pred != null) {
            node.prev = pred;
            //3.cas安全的将当前节点添加到队列末尾
            if (compareAndSetTail(pred, node)) {
                pred.next = node;
                return node;
            }
        }
        //添加失败(有其他节点抢先添加到队列末尾)或此时队列为空
        enq(node);
        return node;
    }

enq(node)方法用来处理阻塞线程并发插入内置队列的情况

private Node enq(final Node node) {
        for (;;) {
            Node t = tail;
            if (t == null) { // Must initialize
                if (compareAndSetHead(new Node()))
                    tail = head;
            } else {
                node.prev = t;
                if (compareAndSetTail(t, node)) {
                    t.next = node;
                    return t;
                }
            }
        }
    }

通过死循环保证节点被正确的添加到队列末尾,同时每次进入循环时会拿到最新的尾节点

acquireQueued(final Node node, int arg)方法节点自旋获得同步状态

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; // help GC
                    failed = false;
                    return interrupted;
                }
                if (shouldParkAfterFailedAcquire(p, node) &&
                    parkAndCheckInterrupt())
                    interrupted = true;
            }
        } finally {
            if (failed)
                cancelAcquire(node);
        }
    }

通过死循环做到自旋
如果前驱节点是头节点并且成功获得同步状态,则当前节点将变成头节点

2.独占式释放锁

 public final boolean release(int arg) {
        if (tryRelease(arg)) {
            Node h = head;
            if (h != null && h.waitStatus != 0)
                unparkSuccessor(h);
            return true;
        }
        return false;
    }

模板方法release(int arg)调用可重写方法tryRelease(arg),并且释放是从头节点开始释放的,释放通过调用LockSupport中的unparkSuccessor(h)方法实现的

共享式模式核心方法源码分析
1.共享式获得锁

public final void acquireShared(int arg) {
        if (tryAcquireShared(arg) < 0)
            doAcquireShared(arg);
    }

如果获得锁失败,将调用doAcquireShared(arg)方法将线程和同步状态构成节点加入到队列中,tryAcquireShared(arg)为可重写的方法,如果返回值大于0则代表成功获得了锁

private void doAcquireShared(int arg) {
        final Node node = addWaiter(Node.SHARED);
        boolean failed = true;
        try {
            boolean interrupted = false;
            //进行自旋获得同步状态
            for (;;) {
                final Node p = node.predecessor();
                if (p == head) {
                    int r = tryAcquireShared(arg);
                    if (r >= 0) {
                        setHeadAndPropagate(node, r);
                        p.next = null; // help GC
                        if (interrupted)
                            selfInterrupt();
                        failed = false;
                        return;
                    }
                }
                if (shouldParkAfterFailedAcquire(p, node) &&
                    parkAndCheckInterrupt())
                    interrupted = true;
            }
        } finally {
            if (failed)
                cancelAcquire(node);
        }
    }

共享式获得锁的实现思路和独占式获得锁的实现思路类似,有部分代码是公用的,不同处为1)独占式通过boolean值来判断线程是否获得同步状态,而共享式通过返回值是否大于0来判断是否获得锁 2)独占式中通过acquireQueued方法来实现自旋,而在共享式中将实现直接写在了doAcquireShared

2.共享式释放锁

 public final boolean releaseShared(int arg) {
        if (tryReleaseShared(arg)) {
            doReleaseShared();
            return true;
        }
        return false;
    }

doReleaseShared()方法的实现

private void doReleaseShared() {
        for (;;) {
            Node h = head;
            if (h != null && h != tail) {
                int ws = h.waitStatus;
                if (ws == Node.SIGNAL) {
                    if (!compareAndSetWaitStatus(h, Node.SIGNAL, 0))
                        continue;            // loop to recheck cases
                    unparkSuccessor(h);
                }
                else if (ws == 0 &&
                         !compareAndSetWaitStatus(h, 0, Node.PROPAGATE))
                    continue;                // loop on failed CAS
            }
            if (h == head)                   // loop if head changed
                break;
        }
    }

共享式释放锁的步骤可以分为两步
1)调用tryReleaseShared方法释放同步状态
2)调用doReleaseShared释放头节点
共享式中为了保证安全释放节点,通过for循环和cas来实现节点的释放

独占式超时获取同步状态

 private boolean doAcquireNanos(int arg, long nanosTimeout)
            throws InterruptedException {
        if (nanosTimeout <= 0L)
            return false;
        //1.计算deadline时间
        final long deadline = System.nanoTime() + nanosTimeout;
        final Node node = addWaiter(Node.EXCLUSIVE);
        boolean failed = true;
        try {
            for (;;) {
                final Node p = node.predecessor();
                if (p == head && tryAcquire(arg)) {
                    setHead(node);
                    p.next = null; // help GC
                    failed = false;
                    return true;
                }
                //2.每次进入循环执行相应的操作后在计算一下剩余的时间
                nanosTimeout = deadline - System.nanoTime();
               //如果小于0,则超时获取锁失败 
                if (nanosTimeout <= 0L)
                    return false;
                if (shouldParkAfterFailedAcquire(p, node) &&
                    nanosTimeout > spinForTimeoutThreshold)
                    LockSupport.parkNanos(this, nanosTimeout);
                if (Thread.interrupted())
                    throw new InterruptedException();
            }
        } finally {
            if (failed)
                cancelAcquire(node);
        }
    }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,591评论 6 501
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,448评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,823评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,204评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,228评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,190评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,078评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,923评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,334评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,550评论 2 333
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,727评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,428评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,022评论 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,672评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,826评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,734评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,619评论 2 354

推荐阅读更多精彩内容