Java CyclicBarrier和CountDownLatch的区别

简介: Java CyclicBarrier和CountDownLatch的区别

Java CyclicBarrier和CountDownLatch的区别

在Java中,CyclicBarrier和CountDownLatch是用于多线程协作的工具类,它们都可以用于线程间的同步和等待,但在使用方式和场景上有一些区别。

CyclicBarrier(循环屏障):

  • CyclicBarrier用于让多个线程相互等待,直到达到某个公共的屏障点,然后同时开始执行下一阶段的任务。
  • CyclicBarrier可以重复使用,当所有线程都到达屏障点时,屏障会打开,所有线程可以继续执行后续任务。
  • CyclicBarrier适用于任务分阶段执行的场景,要求多个线程在同一时间点相互等待,并且可以同时开始下一阶段的任务。

代码示例:

CyclicBarrier barrier = new CyclicBarrier(3, () -> {
    // 当所有线程都到达屏障点时执行的任务
    System.out.println("All threads reached the barrier");
});

// 线程1
new Thread(() -> {
    // 线程1的任务
    try {
        Thread.sleep(1000);
        System.out.println("Thread 1 finished");
        barrier.await(); // 等待其他线程到达屏障点
    } catch (InterruptedException | BrokenBarrierException e) {
        e.printStackTrace();
    }
}).start();

// 线程2和线程3类似

// 输出:
// Thread 1 finished
// Thread 2 finished
// Thread 3 finished
// All threads reached the barrier

CountDownLatch(倒计时门闩):

  • CountDownLatch用于实现一个或多个线程等待其他线程执行完毕的场景。
  • CountDownLatch初始化一个计数器,每个线程执行完任务后将计数器减一,等待的线程调用await()方法等待计数器为零时继续执行。
  • CountDownLatch一旦计数器为零,无法重置,所有等待的线程将同时被释放。

代码示例:

CountDownLatch latch = new CountDownLatch(3);

// 线程1
new Thread(() -> {
    // 线程1的任务
    System.out.println("Thread 1 finished");
    latch.countDown(); // 计数器减一
}).start();

// 线程2和线程3类似

try {
    latch.await(); // 等待计数器为零
    System.out.println("All threads finished");
} catch (InterruptedException e) {
    e.printStackTrace();
}

// 输出:
// Thread 1 finished
// Thread 2 finished
// Thread 3 finished
// All threads finished

CyclicBarrier和CountDownLatch的区别:

  • CyclicBarrier适合线程间等待彼此达到屏障点,然后同时开始下一阶段的任务,可以重复使用。而CountDownLatch适合一个或多个线程等待其他线程执行完毕后再继续执行,计数器无法重置。
  • CyclicBarrier更适合多阶段任务的协作,而CountDownLatch更适合某个任务的前置准备或者等待多个任务同时完成的场景。

根据具体的业务需求和协作场景,选择合适的同步工具类可以提高程序的可读性和可维护性。

相关文章
|
21天前
|
Java
Java之CountDownLatch原理浅析
本文介绍了Java并发工具类`CountDownLatch`的使用方法、原理及其与`Thread.join()`的区别。`CountDownLatch`通过构造函数接收一个整数参数作为计数器,调用`countDown`方法减少计数,`await`方法会阻塞当前线程,直到计数为零。文章还详细解析了其内部机制,包括初始化、`countDown`和`await`方法的工作原理,并给出了一个游戏加载场景的示例代码。
Java之CountDownLatch原理浅析
|
9天前
|
存储 缓存 安全
java 中操作字符串都有哪些类,它们之间有什么区别
Java中操作字符串的类主要有String、StringBuilder和StringBuffer。String是不可变的,每次操作都会生成新对象;StringBuilder和StringBuffer都是可变的,但StringBuilder是非线程安全的,而StringBuffer是线程安全的,因此性能略低。
|
26天前
|
Java
Java代码解释++i和i++的五个主要区别
本文介绍了前缀递增(++i)和后缀递增(i++)的区别。两者在独立语句中无差异,但在赋值表达式中,i++ 返回原值,++i 返回新值;在复杂表达式中计算顺序不同;在循环中虽结果相同但使用方式有别。最后通过 `Counter` 类模拟了两者的内部实现原理。
Java代码解释++i和i++的五个主要区别
|
2月前
|
Java
通过Java代码解释成员变量(实例变量)和局部变量的区别
本文通过一个Java示例,详细解释了成员变量(实例变量)和局部变量的区别。成员变量属于类的一部分,每个对象有独立的副本;局部变量则在方法或代码块内部声明,作用范围仅限于此。示例代码展示了如何在类中声明和使用这两种变量。
|
2月前
|
Java
Java基础之 JDK8 HashMap 源码分析(中间写出与JDK7的区别)
这篇文章详细分析了Java中HashMap的源码,包括JDK8与JDK7的区别、构造函数、put和get方法的实现,以及位运算法的应用,并讨论了JDK8中的优化,如链表转红黑树的阈值和扩容机制。
28 1
|
2月前
|
Java
Java代码解释静态代理和动态代理的区别
### 静态代理与动态代理简介 **静态代理**:代理类在编译时已确定,目标对象和代理对象都实现同一接口。代理类包含对目标对象的引用,并在调用方法时添加额外操作。 **动态代理**:利用Java反射机制在运行时生成代理类,更加灵活。通过`Proxy`类和`InvocationHandler`接口实现,无需提前知道接口的具体实现细节。 示例代码展示了两种代理方式的实现,静态代理需要手动创建代理对象,而动态代理通过反射机制自动创建。
|
2月前
|
缓存 算法 Java
Java 中线程和纤程Fiber的区别是什么?
【10月更文挑战第14天】
75 0
|
13天前
|
Java 开发者
Java多线程编程中的常见误区与最佳实践####
本文深入剖析了Java多线程编程中开发者常遇到的几个典型误区,如对`start()`与`run()`方法的混淆使用、忽视线程安全问题、错误处理未同步的共享变量等,并针对这些问题提出了具体的解决方案和最佳实践。通过实例代码对比,直观展示了正确与错误的实现方式,旨在帮助读者构建更加健壮、高效的多线程应用程序。 ####
|
4天前
|
缓存 Java 开发者
Java多线程编程的陷阱与最佳实践####
本文深入探讨了Java多线程编程中常见的陷阱,如竞态条件、死锁和内存一致性错误,并提供了实用的避免策略。通过分析典型错误案例,本文旨在帮助开发者更好地理解和掌握多线程环境下的编程技巧,从而提升并发程序的稳定性和性能。 ####
|
4天前
|
安全 Java 开发者
Java中的多线程编程:从基础到实践
本文深入探讨了Java多线程编程的核心概念和实践技巧,旨在帮助读者理解多线程的工作原理,掌握线程的创建、管理和同步机制。通过具体示例和最佳实践,本文展示了如何在Java应用中有效地利用多线程技术,提高程序性能和响应速度。
26 1