什么时候使用CountDownLatch

正如每个Java文档所描述的那样,CountDownLatch是一个同步工具类,它允许一个或多个线程一直等待,直到其他线程的操作执行完后再执行。在Java并发中,countdownlatch的概念是一个常见的面试题,所以一定要确保你很好的理解了它。在这篇文章中,我将会涉及到在Java并发编 程中跟CountDownLatch相关的以下几点:

目录

  • CountDownLatch是什么?
  • CountDownLatch如何工作?
  • 在实时系统中的应用场景
  • 应用范例
  • 常见的面试题

CountDownLatch是什么

CountDownLatch是在java1.5被引入的,跟它一起被引入的并发工具类还有CyclicBarrier、Semaphore、ConcurrentHashMapBlockingQueue,它们都存在于java.util.concurrent包下。CountDownLatch这个类能够使一个线程等待其他线程完成各自的工作后再执行。例如,应用程序的主线程希望在负责启动框架服务的线程已经启动所有的框架服务之后再执行。

CountDownLatch是通过一个计数器来实现的,计数器的初始值为线程的数量。每当一个线程完成了自己的任务后,计数器的值就会减1。当计数器值到达0时,它表示所有的线程已经完成了任务,然后在闭锁上等待的线程就可以恢复执行任务。

CountDownLatch的伪代码如下所示:

1

2

3

4

5

6

//Main thread start

//Create CountDownLatch for N threads

//Create and start N threads

//Main thread wait on latch

//N threads completes there tasks are returns

//Main thread resume execution

CountDownLatch如何工作

CountDownLatch.java类中定义的构造函数:

1

2

//Constructs a CountDownLatch initialized with the given count.

public void CountDownLatch(int count) {...}

构造器中的计数值(count)实际上就是闭锁需要等待的线程数量。这个值只能被设置一次,而且CountDownLatch没有提供任何机制去重新设置这个计数值

与CountDownLatch的第一次交互是主线程等待其他线程。主线程必须在启动其他线程后立即调用CountDownLatch.await()方法。这样主线程的操作就会在这个方法上阻塞,直到其他线程完成各自的任务。

其他N 个线程必须引用闭锁对象,因为他们需要通知CountDownLatch对象,他们已经完成了各自的任务。这种通知机制是通过 CountDownLatch.countDown()方法来完成的;每调用一次这个方法,在构造函数中初始化的count值就减1。所以当N个线程都调 用了这个方法,count的值等于0,然后主线程就能通过await()方法,恢复执行自己的任务。

在实时系统中的使用场景

让我们尝试罗列出在java实时系统中CountDownLatch都有哪些使用场景。我所罗列的都是我所能想到的。如果你有别的可能的使用方法,请在留言里列出来,这样会帮助到大家。

  1. 实现最大的并行性:有时我们想同时启动多个线程,实现最大程度的并行性。例如,我们想测试一个单例类。如果我们创建一个初始计数为1的CountDownLatch,并让所有线程都在这个锁上等待,那么我们可以很轻松地完成测试。我们只需调用 一次countDown()方法就可以让所有的等待线程同时恢复执行。
  2. 开始执行前等待n个线程完成各自任务:例如应用程序启动类要确保在处理用户请求前,所有N个外部系统已经启动和运行了。
  3. 死锁检测:一个非常方便的使用场景是,你可以使用n个线程访问共享资源,在每次测试阶段的线程数目是不同的,并尝试产生死锁。

CountDownLatch使用例子

在这个例子中,我模拟了一个应用程序启动类,它开始时启动了n个线程类,这些线程将检查外部系统并通知闭锁,并且启动类一直在闭锁上等待着。一旦验证和检查了所有外部服务,那么启动类恢复执行。

BaseHealthChecker.java:这个类是一个Runnable,负责所有特定的外部服务健康的检测。它删除了重复的代码和闭锁的中心控制代码。

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

public abstract class BaseHealthChecker implements Runnable {

 

    private CountDownLatch _latch;

    private String _serviceName;

    private boolean _serviceUp;

 

    //Get latch object in constructor so that after completing the task, thread can countDown() the latch

    public BaseHealthChecker(String serviceName, CountDownLatch latch)

    {

        super();

        this._latch = latch;

        this._serviceName = serviceName;

        this._serviceUp = false;

    }

 

    @Override

    public void run() {

        try {

            verifyService();

            _serviceUp = true;

        } catch (Throwable t) {

            t.printStackTrace(System.err);

            _serviceUp = false;

        } finally {

            if(_latch != null) {

                _latch.countDown();

            }

        }

    }

 

    public String getServiceName() {

        return _serviceName;

    }

 

    public boolean isServiceUp() {

        return _serviceUp;

    }

    //This methos needs to be implemented by all specific service checker

    public abstract void verifyService();

}

NetworkHealthChecker.java:这个类继承了BaseHealthChecker,实现了verifyService()方法。DatabaseHealthChecker.javaCacheHealthChecker.java除了服务名和休眠时间外,与NetworkHealthChecker.java是一样的。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

public class NetworkHealthChecker extends BaseHealthChecker

{

    public NetworkHealthChecker (CountDownLatch latch)  {

        super("Network Service", latch);

    }

 

    @Override

    public void verifyService()

    {

        System.out.println("Checking " + this.getServiceName());

        try

        {

            Thread.sleep(7000);

        }

        catch (InterruptedException e)

        {

            e.printStackTrace();

        }

        System.out.println(this.getServiceName() + " is UP");

    }

}

ApplicationStartupUtil.java:这个类是一个主启动类,它负责初始化闭锁,然后等待,直到所有服务都被检测完。

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

46

47

48

49

50

51

52

public class ApplicationStartupUtil

{

    //List of service checkers

    private static List<BaseHealthChecker> _services;

 

    //This latch will be used to wait on

    private static CountDownLatch _latch;

 

    private ApplicationStartupUtil()

    {

    }

 

    private final static ApplicationStartupUtil INSTANCE = new ApplicationStartupUtil();

 

    public static ApplicationStartupUtil getInstance()

    {

        return INSTANCE;

    }

 

    public static boolean checkExternalServices() throws Exception

    {

        //Initialize the latch with number of service checkers

        _latch = new CountDownLatch(3);

 

        //All add checker in lists

        _services = new ArrayList<BaseHealthChecker>();

        _services.add(new NetworkHealthChecker(_latch));

        _services.add(new CacheHealthChecker(_latch));

        _services.add(new DatabaseHealthChecker(_latch));

 

        //Start service checkers using executor framework

        Executor executor = Executors.newFixedThreadPool(_services.size());

 

        for(final BaseHealthChecker v : _services)

        {

            executor.execute(v);

        }

 

        //Now wait till all services are checked

        _latch.await();

 

        //Services are file and now proceed startup

        for(final BaseHealthChecker v : _services)

        {

            if( ! v.isServiceUp())

            {

                return false;

            }

        }

        return true;

    }

}

现在你可以写测试代码去检测一下闭锁的功能了。

1

2

3

4

5

6

7

8

9

10

11

12

public class Main {

    public static void main(String[] args)

    {

        boolean result = false;

        try {

            result = ApplicationStartupUtil.checkExternalServices();

        } catch (Exception e) {

            e.printStackTrace();

        }

        System.out.println("External services validation completed !! Result was :: "+ result);

    }

}

1

2

3

4

5

6

7

8

9

Output in console:

 

Checking Network Service

Checking Cache Service

Checking Database Service

Database Service is UP

Cache Service is UP

Network Service is UP

External services validation completed !! Result was :: true

常见面试题

可以为你的下次面试准备以下一些CountDownLatch相关的问题:

  • 解释一下CountDownLatch概念?
  • CountDownLatch 和CyclicBarrier的不同之处?
  • 给出一些CountDownLatch使用的例子?
  •  CountDownLatch 类中主要的方法?
`CountDownLatch` 是 Java 并发包 (`java.util.concurrent`) 提供的一种同步工具类,用于让一个或多个线程等待其他线程完成操作后再继续执行。它通过计数器(count)来进行控制,当计数器减至零时,所有因等待而阻塞的线程会被唤醒并继续运行。 --- ### 核心功能 1. **初始化**:通过构造函数设置初始值 (即需要等待的任务数量)。 2. **倒计时**:每个任务完成后调用 `countDown()` 方法使计数器递减。 3. **等待结束**:主线程或其他线程可以调用 `await()` 或带超时时间的 `await(long timeout, TimeUnit unit)` 方法来等待直到计数值归零。 一旦计数到达零之后,再对 latch 进行额外 countdown 操作不会引发异常也无效;此外已经开启通行状态后的 await 呼叫会立即返回不再有任何延迟效果。 --- ### 示例代码 下面的例子展示了如何使用 `CountDownLatch` 让主程序等待若干个子线程全部执行完毕才退出: ```java import java.util.concurrent.CountDownLatch; public class CountDownLatchExample { public static void main(String[] args) throws InterruptedException { int numberOfThreads = 3; CountDownLatch latch = new CountDownLatch(numberOfThreads); for(int i=1;i<=numberOfThreads ;i++) { Thread worker = new WorkerThread(latch,"Worker-"+i); worker.start(); } System.out.println("Waiting for all threads to finish..."); // 主线程在此处等待,直至所有工作线程完成其任务 latch.await(); System.out.println("All tasks completed."); } } class WorkerThread extends Thread{ private final CountDownLatch latch; public WorkerThread(CountDownLatch latch,String name){ super(name); this.latch=latch; } @Override public void run(){ try{ // Simulate task processing time. sleep((long)(Math.random()*1000)); System.out.println(Thread.currentThread().getName()+" finished."); }finally{ // Signal that we're done with our work so the count can be decremented by one. latch.countDown(); } } } ``` --- ### 输出示例 ``` Waiting for all threads to finish... Worker-1 finished. Worker-2 finished. Worker-3 finished. All tasks completed. ``` 在这个例子中,我们创建了三个工作者线程并且设置了相应的倒计时锁存器实例。主线程在开始的时候打印了一条消息表示正在等待所有的子线程都完成了它们的工作,在每一个子线程结束了它的模拟计算后都会去通知一次这个锁存器对象减少他的内部计数器数值,最后当所有的子线程都已经成功地告知他们已完成各自的职责之时,则原来处于挂起状态下的main thread 就得以恢复正常的运作轨迹并显示出最终的消息表明一切皆已就绪。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值