美文网首页
CountDownLatch分析

CountDownLatch分析

作者: 炫迈哥 | 来源:发表于2017-04-19 21:38 被阅读0次

1.类声明

算是最简单最简单的一个同步工具类了。。。哈哈哈

// 没有继承类,没有实现接口
public class CountDownLatch {
}

2.核心执行依旧靠Sync内部类(基本上所有工具类,包括Lock都是这个模式实现)

private static final class Sync extends AbstractQueuedSynchronizer {
        private static final long serialVersionUID = 4982264981922014374L;

        Sync(int count) {
            // 设置AQS的资源数量
            setState(count);
        }

        int getCount() {
            return getState();
        }

        // 非常重要的实现,只有当当前资源数量为0的时候才算是获取到锁
        protected int tryAcquireShared(int acquires) {
            return (getState() == 0) ? 1 : -1;
        }
        // 以共享模式释放一个资源, 哈哈这里很好玩,int releases是AQS生命的函数带有的参数,这里根本没用它,因为每次Countdown本来就只减一,但是感觉怪怪的。
        protected boolean tryReleaseShared(int releases) {
            // Decrement count; signal when transition to zero
            for (;;) { // 比较简单的CAS递减操作。
                int c = getState();
                if (c == 0)  // 已经是0了,直接返回false,AQS什么也不干
                    return false; 
                int nextc = c-1;
                if (compareAndSetState(c, nextc))
                    return nextc == 0; // 操作完成后资源数为0则返回true,不为0返回false,AQS什么也不干
            }
        }
    }

3.构造函数

public CountDownLatch(int count) {
        // 可以看到初始值不能小于0
        if (count < 0) throw new IllegalArgumentException("count < 0");
        // 实例化一个Sync
        this.sync = new Sync(count);
    }

4.await方法

public void await() throws InterruptedException {
        // 调用了AQS的尝试获取共享锁可中断的方法,因为
        // Sync在实现tryAcquireShared时,是以getState是否为0来判断获取到锁的,所以
        // 只要当前state还不为0,await县城将会被放进等待队列。
        sync.acquireSharedInterruptibly(1);
    }
// 调用了支持超时的获取锁方法
public boolean await(long timeout, TimeUnit unit)
        throws InterruptedException {
        return sync.tryAcquireSharedNanos(1, unit.toNanos(timeout));
    }

AQS中acquireSharedInterruptibly方法

public final void acquireSharedInterruptibly(int arg)
            throws InterruptedException {
        if (Thread.interrupted())
            throw new InterruptedException();
        if (tryAcquireShared(arg) < 0) // (getState() == 0) ? 1 : -1;小于0代表state还大于0
            doAcquireSharedInterruptibly(arg);
    }

5.countDown方法

public void countDown() {
       // 调用AQS的释放共享资源方法
       sync.releaseShared(1);
   }

AQS的releaseShared源码

public final boolean releaseShared(int arg) {
// 当前扣减后剩余量为0时,tryReleaseShared返回true,将会执行doReleaseShared唤醒await线程
// 就是说新建闭锁时初始化资源数为5,但是程序中有6个线程去做countDown,那么最后一个countDown的线程将不会造成任何影响
        if (tryReleaseShared(arg)) { 
            doReleaseShared();
            return true;
        }
        return false;
    }

相关文章

  • 3、concurrent包下常用的辅助类

    CountDownLatch:(减法计数器) 示例: 结果: 分析:countDownLatch.countDow...

  • 并发工具之 CountDownLatch

    目录 -CountDownLatch 是什么?-CountDownLatch 用法-源码分析-应用范例 Count...

  • CountDownLatch 分析

    参考 一行一行源码分析清楚AQSAQS 独占锁:只能有一个线程持有锁,获取锁失败的线程进入阻塞队列,持有锁的线程释...

  • CountDownLatch分析

    1.类声明 算是最简单最简单的一个同步工具类了。。。哈哈哈 2.核心执行依旧靠Sync内部类(基本上所有工具类,包...

  • CountDownLatch 源码分析

    1. 类介绍 一个同步辅助类,在完成一组正在其他线程中执行的操作之前,它允许一个或多个线程一直等待。用给定的计数 ...

  • CountDownLatch源码分析

    CountDownLatch 定义CountDownLatch 可以理解为一个简单的计数器, 首先设定计数的值, ...

  • countdownlatch源码分析

    countdownlatch是JDK提供的一个线程控制的工具类,虽然代码短少,实现简单,但是它的作用却十分的大。 ...

  • CountDownLatch源码分析

    并发源码分析篇: ReentrantLock源码分析 ReentrantReadWriteLock源码分析 Con...

  • CountDownLatch源码分析

    实例代码 构造函数new CountDownLatch(3) countDownLatch.countDown()...

  • CountDownLatch源码分析

    所需知识储备:队列、AQS、自旋锁、线程池等。如果对AbstractQueeuedSynchronizer(AQS...

网友评论

      本文标题:CountDownLatch分析

      本文链接:https://www.haomeiwen.com/subject/gtkhzttx.html