CountDownLatch源码解析
发布时间: 2024-01-10 14:23:07 阅读量: 54 订阅数: 34
CountDownLatch源码解析之await()
# 1. 引言
### 1.1 简介
CountDownLatch是Java并发包中的一个实用工具类,用于实现线程的等待和同步操作。它可以让某个线程等待其他线程执行完毕后再继续执行,起到线程协作的作用。
### 1.2 作用和特点
CountDownLatch的主要作用是帮助线程在一组并发任务完成之前进行等待,然后再继续执行。它的特点可以总结为以下几点:
- 可以控制线程的执行顺序,实现线程间的协作。
- 可以实现线程的等待和同步操作。
- 可以设置等待的超时时间,防止无限等待。
- 简单易用,适用于各种场景。
在接下来的章节中,我们将深入了解CountDownLatch的基本使用、底层实现原理、常见应用场景、注意事项和使用技巧,以及与其他同类工具的比较。让我们开始吧!
# 2. CountDownLatch的基本使用
CountDownLatch是一个在多线程环境下非常常用的工具类,用于控制线程的执行顺序。它可以让一个或多个线程等待其他线程完成操作后再执行,具有非常强大的功能。
#### 2.1 变量初始化
在使用CountDownLatch时,首先需要创建一个CountDownLatch对象,并指定初始计数值。这个计数值代表了需要等待完成的线程数量。
在Java中,可以这样初始化一个CountDownLatch:
```java
import java.util.concurrent.CountDownLatch;
public class Main {
public static void main(String[] args) {
int threadCount = 3;
CountDownLatch latch = new CountDownLatch(threadCount);
// 其他线程的逻辑...
}
}
```
在这个例子中,我们初始化了一个CountDownLatch对象`latch`,并将初始计数值设置为3。
#### 2.2 等待线程
接下来,在需要等待的线程中,可以调用`await()`方法来阻塞当前线程,直至CountDownLatch计数值为0。
```java
public class Worker implements Runnable {
private CountDownLatch latch;
public Worker(CountDownLatch latch) {
this.latch = latch;
}
public void run() {
try {
// 模拟线程执行任务的耗时
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("任务执行完成");
latch.countDown(); // 完成任务,计数减一
}
}
public class Main {
public static void main(String[] args) {
int threadCount = 3;
CountDownLatch latch = new CountDownLatch(threadCount);
for (int i = 0; i < threadCount; i++) {
Thread workerThread = new Thread(new Worker(latch));
workerThread.start();
}
// 等待所有任务执行完成
try {
latch.await();
System.out.println("所有任务执行完成");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
```
在这个例子中,我们创建了3个Worker线程,它们会分别执行任务,并在任务完成后调用`countDown()`方法对CountDownLatch的计数进行减一。在主线程中,我们调用了`latch.await()`来等待所有任务执行完成。
#### 2.3 完成信号
CountDownLatch还可以通过`countDown()`方法来发出完成信号,表示已经完成了一定的任务,这样等待中的线程就可以继续执行。
综上所述,CountDownLatch的基本使用涵盖了初始化、等待线程和完成信号的相关操作。
# 3. CountDownLatch的底层实现原理
在了解CountDownLatch的底层实现原理之前,我们先来介绍一下AQS(AbstractQueuedSynchronizer)。
#### 3.1 AQS(AbstractQueuedSynchronizer)介绍
AQS是Java中用于实现各种同步器的基础框架。它提供了一套强大的原子操作和线程阻塞/唤醒机制,可以方便地构建各种高性能的并发工具。
AQS的核心思想是利用一个int类型的volatile变量state来表示同步状态,通过CAS(Compare and Set)操作来实现对state的原子更新。并且通过一个FIFO队列来管理阻塞的线程,确保线程获取同步状态的顺序。
#### 3.2 CountDownLatch源码解析
CountDownLatch的底层实现就是基于AQS来完成的。实际上,CountDownLatch内部持有一个AQS的实例,默认继承了同步器的所有方法。
CountDownLatch的构造函数会初始化一个给定的state值,表示需要等待的线程数量。当调用await方法时,当前线程会尝试获取同步状态,如果state不为0,则会进入阻塞状态;而当调用countDown方法时,会对state进行减1操作,并唤醒阻塞的线程。
下面是CountDownLatch的简化版源码:
```java
public class CountDownLatch {
private final Sync sync;
public CountDownLatch(int count) {
if (count < 0)
throw new IllegalArgumentException("count < 0");
sync = new Sync(count);
}
public void await() throws InterruptedException {
sync.acquireSharedInterruptibly(1);
}
public void countDown() {
sync.releaseShared(1);
}
private static final class Sync extends AbstractQueuedSynchronizer {
Sync(int count) {
setState(count);
}
@Override
protected int tryAcquireShared(int arg) {
return (getState() == 0) ? 1 : -1;
}
@Override
protected boolean tryReleaseShared(int arg) {
for (;;) {
int current = getState();
if (current == 0)
return false;
int next = current - 1;
if (compareAndSetState(current, next))
return next == 0;
}
}
}
}
```
可以看到,CountDownLatch使用了AQS的共享模式,通过重写tryAcquireShared和tryReleaseShared方法来实现共享锁的获取和释放。
### 示例代码
```java
import java.util.concurrent.CountDownLatch;
public class CountDownLatchExample {
public static void main(String[] args) throws InterruptedException {
CountDownLatch latch = new CountDownLatch(3);
// 创建三个线程,并行执行任务
Thread t1 = new Thread(new Task(latch, "A"));
Thread t2 = new Thread(new Task(latch, "B"));
Thread t3 = new Thread(new Task(latch, "C"));
t1.start();
t2.start();
t3.start();
// 等待三个线程任务完成
latch.await();
System.out.println("All tasks completed!");
}
static class Task implements Runnable {
private final CountDownLatch latch;
private final String name;
Task(CountDownLatch latch, String name) {
this.latch = latch;
this.name = name;
}
@Override
public void run() {
try {
System.out.println("Task " + name + " is running");
Thread.sleep(1000);
System.out.println("Task " + name + " is completed");
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
// 任务完成后对计数器减1
latch.countDown();
}
}
}
}
```
代码解释:
- 创建了一个CountDownLatch对象,并设置计数器的初始值为3。
- 创建了3个线程,每个线程执行一个任务,并在任务完成后调用countDown方法,将计数器减1。
- 主线程调用await方法,等待计数器变为0,表示所有任务都已完成。
- 当三个任务全部完成后,主线程得以继续执行。
代码输出:
```
Task A is running
Task B is running
Task C is running
Task A is completed
Task B is completed
Task C is completed
All tasks completed!
```
本章节介绍了CountDownLatch的底层实现原理,通过AQS的同步器来实现等待/唤醒和原子操作。同时给出了一个简单的示例代码来演示CountDownLatch的使用过程。
# 4. CountDownLatch的常见应用场景
在实际的开发中,CountDownLatch有许多常见的应用场景,下面将介绍一些常见的使用方式。
#### 4.1 并发任务的等待
在某些并发任务中,需要等待所有任务执行完成后再进行下一步操作。这时可以利用CountDownLatch来实现等待机制,具体代码如下所示:
```java
import java.util.concurrent.CountDownLatch;
public class ConcurrentTaskDemo {
public static void main(String[] args) {
int taskCount = 5;
CountDownLatch latch = new CountDownLatch(taskCount);
for (int i = 0; i < taskCount; i++) {
new Thread(() -> {
// 执行任务
// ...
latch.countDown(); // 任务执行完成,递减计数器
}).start();
}
try {
latch.await(); // 等待所有任务执行完成
System.out.println("所有任务执行完成,可以进行下一步操作");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
```
在上面的代码中,我们创建了一个CountDownLatch实例,然后在多个线程执行任务时,使用`countDown()`方法递减计数器,主线程通过`await()`方法等待所有任务执行完成后再进行下一步操作。
#### 4.2 等待资源初始化
有时候在系统启动时,需要等待一些资源的初始化完成后再启动其他模块,可以利用CountDownLatch来实现等待初始化完成的机制。下面是一个简单的示例代码:
```java
import java.util.concurrent.CountDownLatch;
public class ResourceInitDemo {
private CountDownLatch initLatch = new CountDownLatch(1);
public void init() {
// 初始化资源
// ...
initLatch.countDown(); // 资源初始化完成,递减计数器
}
public void doSomething() {
try {
initLatch.await(); // 等待资源初始化完成
// 执行需要等待资源初始化的操作
// ...
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
```
在上面的示例中,资源初始化完成后调用`countDown()`方法来递减计数器,而`doSomething()`方法则通过`await()`方法来等待资源初始化完成。
#### 4.3 等待外部服务启动
在一些分布式系统中,某个模块依赖于外部服务的启动,需要等待外部服务启动完成后再启动自身模块。这时可以利用CountDownLatch来实现等待外部服务启动的机制,具体代码如下所示:
```java
import java.util.concurrent.CountDownLatch;
public class ExternalServiceDemo {
public static void main(String[] args) {
CountDownLatch externalServiceLatch = new CountDownLatch(1);
// 启动外部服务的线程
new Thread(() -> {
// 启动外部服务
// ...
externalServiceLatch.countDown(); // 外部服务启动完成,递减计数器
}).start();
try {
externalServiceLatch.await(); // 等待外部服务启动完成
System.out.println("外部服务启动完成,可以启动自身模块");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
```
在上面的代码中,我们使用CountDownLatch来等待外部服务启动完成后再启动自身模块,从而实现了模块间的协同启动。
这些是CountDownLatch的一些常见应用场景,通过合理的使用,可以优雅地处理多线程并发场景下的任务协同问题。
# 5. CountDownLatch的注意事项和使用技巧
在使用CountDownLatch时,有一些注意事项和使用技巧需要牢记,以确保代码的正确性和性能优化。
### 5.1 线程安全性
CountDownLatch是线程安全的,可以同时被多个线程使用。它的内部同步机制确保了线程之间的正确协作。每个线程都可以调用`countDown`方法来主动减少计数器值,并调用`await`方法等待计数器值达到0。一旦计数器值为0,所有等待的线程将被唤醒。在多线程并发的场景下,CountDownLatch可以保证线程之间的同步和协调。
### 5.2 使用场景选择
CountDownLatch可以用于各种场景,但是在选择是否使用CountDownLatch时,需要根据具体的需求来进行评估。CountDownLatch适用于需要等待多个线程完成任务后再继续执行的场景。如果只是需要等待一个线程完成,可以考虑使用`Thread.join()`方法。如果需要实现更复杂的逻辑,可以考虑使用CyclicBarrier或Semaphore。
### 5.3 错误处理和超时机制
在使用CountDownLatch时,需要考虑错误处理和超时机制。如果等待的线程出现异常或意外终止,可能会导致主线程一直等待。为了避免这种情况,建议在等待前设置一个超时时间,如使用`await(long timeout, TimeUnit unit)`方法,如果超过指定的时间仍未等到计数器值达到0,则主动退出等待。
```java
public class Example {
public static void main(String[] args) throws InterruptedException {
CountDownLatch latch = new CountDownLatch(1);
Thread thread = new Thread(() -> {
try {
// 模拟耗时操作
Thread.sleep(3000);
latch.countDown();
} catch (InterruptedException e) {
e.printStackTrace();
}
});
thread.start();
boolean result = latch.await(5, TimeUnit.SECONDS);
if (result) {
System.out.println("CountDownLatch countdown complete.");
} else {
System.out.println("CountDownLatch countdown timeout.");
}
}
}
```
在上面的示例中,主线程等待子线程执行完毕,并设置了超时时间为5秒钟。如果等待时间超过了5秒,将输出"CountDownLatch countdown timeout.";否则,输出"CountDownLatch countdown complete."。
通过合理设置超时时间,可以避免程序一直等待的情况,提高代码的健壮性。
注:以上示例为Java语言的代码示例,其他语言类似,只是具体的语法略有不同。
# 6. CountDownLatch与其他同类工具的比较
在并发编程中,除了CountDownLatch外,还有一些其他的同类工具,它们在不同的场景下有着各自的特点和适用性。接下来,我们将分别比较CountDownLatch与CyclicBarrier以及Semaphore,以及介绍CountDownLatch在分布式系统中的应用。
#### 6.1 CyclicBarrier和CountDownLatch对比
##### 6.1.1 相同点
- 都可以用于线程间的协调和同步
- 都可以实现多个线程等待某个条件达成后再执行
##### 6.1.2 不同点
- CountDownLatch是一次性的,计数器减到0后就无法重置;而CyclicBarrier可以被重置并且可以循环使用
- CyclicBarrier需要所有等待的线程都到达同一个栅栏点时才能继续执行,而CountDownLatch的等待线程可以在计数器归零时立即执行
##### 6.1.3 适用场景
- 当需要一组线程都到达某个同步点时,使用CyclicBarrier更合适
- 当某个线程需要等待其他一组线程都执行完毕后再执行时,使用CountDownLatch更合适
#### 6.2 Semaphore和CountDownLatch对比
##### 6.2.1 相同点
- 都可以用于控制并发线程数量
- 都是基于AQS实现的同步工具
##### 6.2.2 不同点
- Semaphore主要用于控制对特定资源的访问,而CountDownLatch主要用于等待其他线程执行完毕
- Semaphore可以动态调整许可数量,而CountDownLatch的计数一旦归零就无法再次初始化
##### 6.2.3 适用场景
- 当需要控制对一组资源的并发访问时,使用Semaphore更合适
- 当某个线程需要等待其他一组线程都执行完毕后再执行时,使用CountDownLatch更合适
#### 6.3 CountDownLatch在分布式系统中的应用
在分布式系统中,CountDownLatch常用于等待分布式任务的完成。比如,一个分布式任务需要多台服务器协同完成,而某个任务必须等待所有的服务器都完成后才能继续执行,这时就可以使用CountDownLatch来实现这种等待逻辑。在Java中,可以使用Zookeeper等分布式工具来实现分布式的CountDownLatch。
通过以上比较和应用介绍,可以更好地选择合适的工具来满足特定的并发编程需求。
0
0