引言
在并发编程中,CyclicBarrier和CountDownLatch是两个常用的同步工具类。它们都可以用于线程之间的等待和协调,但在使用方式和功能上有一些区别。本文将深入探讨CyclicBarrier和CountDownLatch的区别,并给出相应的代码示例。
CyclicBarrier和CountDownLatch简介
CyclicBarrier
CyclicBarrier是Java提供的一种同步辅助类,它可以让一组线程互相等待,直到到达某个公共屏障点。当所有线程都到达屏障点后,这些线程才能继续执行后续的操作。
CountDownLatch
CountDownLatch也是Java提供的一种同步辅助类,它可以让一个或多个线程等待,直到其他线程执行完一组操作后再继续执行。
区别对比
下面是CyclicBarrier和CountDownLatch的区别对比:
1. 使用方式
CyclicBarrier的使用方式相对复杂一些,需要明确指定线程的数量,并在每个线程到达屏障点后执行特定的操作。
CountDownLatch的使用方式相对简单,只需要调用await()
方法等待其他线程执行完一组操作即可。
2. 是否可重用
CyclicBarrier可以重用,即在所有线程到达屏障点后,屏障会自动重置,线程可以继续使用。
CountDownLatch不能重用,一旦计数器减到0,就无法再次使用。
3. 等待方式
CyclicBarrier等待线程的方式是通过await()
方法进行等待,可以选择等待一定的时间或者等待其他线程到达。
CountDownLatch等待线程的方式是通过await()
方法进行等待,直到计数器减到0。
4. 功能
CyclicBarrier主要用于线程之间的等待和协调,在所有线程到达屏障点后可以执行特定的操作。
CountDownLatch主要用于等待其他线程执行完一组操作后再继续执行。
代码示例
下面给出一个简单的CyclicBarrier和CountDownLatch的代码示例:
import java.util.concurrent.BrokenBarrierException;
import java.util.concurrent.CyclicBarrier;
import java.util.concurrent.CountDownLatch;
public class CyclicBarrierDemo {
public static void main(String[] args) throws InterruptedException {
final int threadCount = 5;
final CyclicBarrier cyclicBarrier = new CyclicBarrier(threadCount, () -> {
System.out.println("所有线程已到达屏障点");
});
for (int i = 0; i < threadCount; i++) {
final int threadNum = i;
new Thread(() -> {
try {
System.out.println("线程" + threadNum + "已到达屏障点");
cyclicBarrier.await();
System.out.println("线程" + threadNum + "继续执行");
} catch (InterruptedException | BrokenBarrierException e) {
e.printStackTrace();
}
}).start();
}
}
}
public class CountDownLatchDemo {
public static void main(String[] args) throws InterruptedException {
final int threadCount = 5;
final CountDownLatch countDownLatch = new CountDownLatch(threadCount);
for (int i = 0; i < threadCount; i++) {
final int threadNum = i;
new Thread(() -> {
try {
System.out.println("线程" + threadNum + "开始执行");
Thread.sleep(1000);
System.out.println("线程" + threadNum + "执行完毕");
countDownLatch.countDown();
} catch (InterruptedException e) {
e.printStackTrace();
}
}).start();
}
countDownLatch.await();
System.out.println("所有线程执行完毕");
}
}