JAVA并发编程系列(8)CountDownLatch核心原理

本文涉及的产品
注册配置 MSE Nacos/ZooKeeper,118元/月
服务治理 MSE Sentinel/OpenSergo,Agent数量 不受限
云原生网关 MSE Higress,422元/月
简介: 面试中的编程题目“模拟拼团”,我们通过使用CountDownLatch来实现多线程条件下的拼团逻辑。此外,深入解析了CountDownLatch的核心原理及其内部实现机制,特别是`await()`方法的具体工作流程。通过详细分析源码与内部结构,帮助读者更好地理解并发编程的关键概念。

拼多多 D2面试,现场编程模拟拼团,10人拼团成功。限时2分钟!开始吧.....!

在面试过程经常有算法题、模拟现实案例、经典功能设计、核心原理分析等。这些看似简单,实际需要候选人有非常扎实的基础,才能应付这些八股考古面试。


和之前文章一样,我们通过现实案例出发,最后抛出主角的方式,带大家由浅入深地了解并发编程核心知识点。


一、面试真题:模拟拼团


    我们利用CountDownLatch倒计时的特性,多线程并发条件下,多线程可以调用CountDownLatch.countDown()方法进行减1,然后等候信号的线程调用CountDownLatch.await()方法,等待CountDownLatch倒数为0,会被唤醒继续执行。


package lading.java.mutithread;
import java.util.HashSet;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
/**
 * 模拟拼团,满10人成团
 */
public class Demo009CountDownLatch {
    public static int total = 10;//成团人数
    public static CountDownLatch buySuccess = new CountDownLatch(total);//倒数门闩
    public static HashSet<String> customersName = new HashSet<>();
    public static void main(String[] args) throws InterruptedException {
        //10个线程,模拟10个客户参团,参团后CountDownLatch.countDown();
        for (int i = 0; i < total; i++) {
            new Thread(() -> {
                try {
                    Thread.sleep(3000);
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
                System.out.println(Thread.currentThread().getName());
                buySuccess.countDown();//参团后,计数减一
                customersName.add(Thread.currentThread().getName());
            }, "客户" + (i + 1) + "参团").start();
        }
        //主线程进行超限等待阻塞,如果CountDownLatch值为0,会被唤醒;如果超时也会唤醒继续执行
        buySuccess.await(5, TimeUnit.SECONDS);//限时5秒,5秒不成团就超时
        //判断是否参团成功
        if (customersName.size() == total) {
            System.out.println("拼团成功,参团客户有:" + customersName);
        } else {
            System.out.println("拼团失败,参团客户数量不足" + total + ",目前参数人数为:" + customersName);
        }
    }
}


二、说说CountDownLatch的核心原理


      看了CountDownLatch源码,发现这个也是JUC家族兄弟,和之前说的《Semaphore信号量剖析》《ReentrantLock核心原理剖析》有非常多相似的地方,都是AQS实现。AQS原理我们之前也专门说过《AQS原理剖析》,以及AQS底层通过CAS的实现,这个《JUC包之CAS原理》也有详细剖析。

      先看一下CountDownLatch类结构图,是JUC包里几个兄弟里代码最少最简单的一个。



      其中唯一的内部类Sync,实现了AQS队列同步器。AQS里面的核心变量volatile int state,就是个共享变量。new CountDownLatch(count )构造器变量count实际就是AQS的state。通过多线程countDown()去修改state值,达到多线程协同效果。



CountDownLatch核心方法就2个,非常简单:


await():调用该方法的线程进行阻塞,等待count值为0被唤醒,继续执行。以及可以设置超时时间,超时后,该阻塞线程就会重行执行。


countDown():就是减一。源码如下,获取到state值后,通过CAS去减1.里面没有竞争锁的逻辑,也没有公平锁、非公平锁这些。


protected boolean tryReleaseShared(int releases) {
            // Decrement count; signal when transition to zero
            for (;;) {
                int c = getState();
                if (c == 0)
                    return false;
                int nextc = c-1;
                if (compareAndSetState(c, nextc))
                    return nextc == 0;
            }
        }
    }


三、说说CountDownLatch的await()方法是如何实现的


大佬问的很细。确实整个CountDownLatch核心的核心就是await(),方法。那个countDown()实在没啥好说的。

我们具体总结一下await():

1、先判断线程是否已中断,如果中断就抛出线程中断异常。


//1 await()方法
    public void await() throws InterruptedException {
        sync.acquireSharedInterruptibly(1);
    }
    
   //2 await()里面的acquireSharedInterruptibly()
    public final void acquireSharedInterruptibly(int arg)
            throws InterruptedException {
        if (Thread.interrupted())
            throw new InterruptedException();
        if (tryAcquireShared(arg) < 0)
            doAcquireSharedInterruptibly(arg);
    }


2、判断state值是否为0,如果是0,那就继续运行。

protected int tryAcquireShared(int acquires) {
      return (getState() == 0) ? 1 : -1;
}


3、如果不是0,就要做线程阻塞等待的准备。具体如下:

先构建一个共享模式NODE节点,并把它放到AQS的FIFO队列里。

然后开始自旋,不断判断当前FIFO队列里,自己是否为头节点,以及判断state共享变量是否为0.就干这点事。


private void doAcquireInterruptibly(int arg)
        throws InterruptedException {
        final Node node = addWaiter(Node.EXCLUSIVE);
        boolean failed = true;
        try {
            for (;;) {
                final Node p = node.predecessor();
                //如果自己是头节点,且state值是0,说明CountDown的倒计时已经为0,不用再等了。
                if (p == head && tryAcquire(arg)) {
                    setHead(node);
                    p.next = null; // help GC
                    failed = false;
                    return;
                }
                //判断是否要挂起当前线程
                if (shouldParkAfterFailedAcquire(p, node) &&
                    parkAndCheckInterrupt())
                    throw new InterruptedException();
            }
        } finally {
            if (failed)
            //如果出现失败或者异常,就取消该节点,唤醒后续节点
                cancelAcquire(node);
        }
    }

今天就分享到这,明天我们分享CyclicBarrier。

相关文章
|
1月前
|
Java 程序员
Java编程中的异常处理:从基础到高级
在Java的世界中,异常处理是代码健壮性的守护神。本文将带你从异常的基本概念出发,逐步深入到高级用法,探索如何优雅地处理程序中的错误和异常情况。通过实际案例,我们将一起学习如何编写更可靠、更易于维护的Java代码。准备好了吗?让我们一起踏上这段旅程,解锁Java异常处理的秘密!
|
10天前
|
存储 缓存 Java
Java 并发编程——volatile 关键字解析
本文介绍了Java线程中的`volatile`关键字及其与`synchronized`锁的区别。`volatile`保证了变量的可见性和一定的有序性,但不能保证原子性。它通过内存屏障实现,避免指令重排序,确保线程间数据一致。相比`synchronized`,`volatile`性能更优,适用于简单状态标记和某些特定场景,如单例模式中的双重检查锁定。文中还解释了Java内存模型的基本概念,包括主内存、工作内存及并发编程中的原子性、可见性和有序性。
Java 并发编程——volatile 关键字解析
|
14天前
|
算法 Java 调度
java并发编程中Monitor里的waitSet和EntryList都是做什么的
在Java并发编程中,Monitor内部包含两个重要队列:等待集(Wait Set)和入口列表(Entry List)。Wait Set用于线程的条件等待和协作,线程调用`wait()`后进入此集合,通过`notify()`或`notifyAll()`唤醒。Entry List则管理锁的竞争,未能获取锁的线程在此排队,等待锁释放后重新竞争。理解两者区别有助于设计高效的多线程程序。 - **Wait Set**:线程调用`wait()`后进入,等待条件满足被唤醒,需重新竞争锁。 - **Entry List**:多个线程竞争锁时,未获锁的线程在此排队,等待锁释放后获取锁继续执行。
49 12
|
11天前
|
监控 Java API
探索Java NIO:究竟在哪些领域能大显身手?揭秘原理、应用场景与官方示例代码
Java NIO(New IO)自Java SE 1.4引入,提供比传统IO更高效、灵活的操作,支持非阻塞IO和选择器特性,适用于高并发、高吞吐量场景。NIO的核心概念包括通道(Channel)、缓冲区(Buffer)和选择器(Selector),能实现多路复用和异步操作。其应用场景涵盖网络通信、文件操作、进程间通信及数据库操作等。NIO的优势在于提高并发性和性能,简化编程;但学习成本较高,且与传统IO存在不兼容性。尽管如此,NIO在构建高性能框架如Netty、Mina和Jetty中仍广泛应用。
26 3
|
11天前
|
安全 算法 Java
Java CAS原理和应用场景大揭秘:你掌握了吗?
CAS(Compare and Swap)是一种乐观锁机制,通过硬件指令实现原子操作,确保多线程环境下对共享变量的安全访问。它避免了传统互斥锁的性能开销和线程阻塞问题。CAS操作包含三个步骤:获取期望值、比较当前值与期望值是否相等、若相等则更新为新值。CAS广泛应用于高并发场景,如数据库事务、分布式锁、无锁数据结构等,但需注意ABA问题。Java中常用`java.util.concurrent.atomic`包下的类支持CAS操作。
43 2
|
11天前
|
存储 安全 Java
Java多线程编程秘籍:各种方案一网打尽,不要错过!
Java 中实现多线程的方式主要有四种:继承 Thread 类、实现 Runnable 接口、实现 Callable 接口和使用线程池。每种方式各有优缺点,适用于不同的场景。继承 Thread 类最简单,实现 Runnable 接口更灵活,Callable 接口支持返回结果,线程池则便于管理和复用线程。实际应用中可根据需求选择合适的方式。此外,还介绍了多线程相关的常见面试问题及答案,涵盖线程概念、线程安全、线程池等知识点。
92 2
|
27天前
|
安全 算法 Java
Java多线程编程中的陷阱与最佳实践####
本文探讨了Java多线程编程中常见的陷阱,并介绍了如何通过最佳实践来避免这些问题。我们将从基础概念入手,逐步深入到具体的代码示例,帮助开发者更好地理解和应用多线程技术。无论是初学者还是有经验的开发者,都能从中获得有价值的见解和建议。 ####
|
27天前
|
Java 调度
Java中的多线程编程与并发控制
本文深入探讨了Java编程语言中多线程编程的基础知识和并发控制机制。文章首先介绍了多线程的基本概念,包括线程的定义、生命周期以及在Java中创建和管理线程的方法。接着,详细讲解了Java提供的同步机制,如synchronized关键字、wait()和notify()方法等,以及如何通过这些机制实现线程间的协调与通信。最后,本文还讨论了一些常见的并发问题,例如死锁、竞态条件等,并提供了相应的解决策略。
50 3
|
Java Android开发
【Java 虚拟机原理】Java 引用类型 ( 强引用 | 软引用 | 弱引用 | 虚引用 | 静态变量 )
【Java 虚拟机原理】Java 引用类型 ( 强引用 | 软引用 | 弱引用 | 虚引用 | 静态变量 )
159 0
|
8天前
|
Java
Java—多线程实现生产消费者
本文介绍了多线程实现生产消费者模式的三个版本。Version1包含四个类:`Producer`(生产者)、`Consumer`(消费者)、`Resource`(公共资源)和`TestMain`(测试类)。通过`synchronized`和`wait/notify`机制控制线程同步,但存在多个生产者或消费者时可能出现多次生产和消费的问题。 Version2将`if`改为`while`,解决了多次生产和消费的问题,但仍可能因`notify()`随机唤醒线程而导致死锁。因此,引入了`notifyAll()`来唤醒所有等待线程,但这会带来性能问题。
Java—多线程实现生产消费者