Java中几种并发容器的使用

简介: Java中几种并发容器的使用

文章目录

一、并发容器

CountDownLatch

CountDownLatch它可以实现类似计数器的功能,当前线程需要等待其他任务执行完成才开始执行。

   /**

    * 构造函数初始化计数的数量,需要等待执行的任务个数

    */

   public CountDownLatch(int count) {

       if (count < 0) throw new IllegalArgumentException("count < 0");

       this.sync = new Sync(count);

   }

   /**

    * 任务执行完毕之后,减去1,当等待任务为0时,唤醒其他线程。

    */

 

public void countDown() {

       sync.releaseShared(1);

   }

    /**

    * 当前线程等到线程的计数器为0时候开始继续执行。

    */

public void await() throws InterruptedException {

       sync.acquireSharedInterruptibly(1);

   }

示例

public class CountDownLatchDemo {

   public static void main(String[] args) throws InterruptedException {

       final CountDownLatch latch = new CountDownLatch(1);

       new Thread(()->{

           System.out.println(Thread.currentThread().getName()+"执行开始....");

           try {

               Thread.sleep(3);

           } catch (InterruptedException e) {

               throw new RuntimeException(e);

           }

           System.out.println(Thread.currentThread().getName()+"执行结束....");

           latch.countDown();

       },"t1").start();

       System.out.println("等待t1线程执行完毕...");

       latch.await();

       System.out.println("t1线程已经执行完毕");

       System.out.println("继续执行主线程");

   }

}

注释掉latch.await(),输出

等待t1线程执行完毕...

t1线程已经执行完毕

继续执行主线程

t1执行开始....

t1执行结束....

加上latch.await(),输出

等待t1线程执行完毕...

t1执行开始....

t1执行结束....

t1线程已经执行完毕

继续执行主线程

CyclicBarrier (回环栅栏)

CyclicBarrier它的作用就是会让所有线程都等待完成后才会继续下一步行

动。

构造器

/**

CyclicBarrier初始化时规定一个数目,

然后计算调用了CyclicBarrier.await()进入等待的线程数。

当线程数达到了这个数目时,所有进入等待状态的线程被唤醒并继续。

*/

  public CyclicBarrier(int parties) {

       this(parties, null);

   }

CyclicBarrier 中最重要的方法就是await 方法:

/**

用来挂起当前线程,直至所有线程都到达barrier 状态再同时执行后续任

务;

*/

public int await() throws InterruptedException, BrokenBarrierException {

       try {

           return dowait(false, 0L);

       } catch (TimeoutException toe) {

           throw new Error(toe); // cannot happen

       }

   }

示例

public class CyclicBarrierDemo {

   public static void main(String[] args) {

       //模拟5个人聚会,先后时间到达,人员到齐之后才可以进行活动

       int N =5;

       CyclicBarrier barrier = new CyclicBarrier(N);

       for(int i=1;i<=N;i++) {

           new Person("person" + i, barrier).start();

       }

   }

   static class Person extends Thread {

       private CyclicBarrier cyclicBarrier;

       public Person(String personname,CyclicBarrier cyclicBarrier) {

           super(personname);

           this.cyclicBarrier = cyclicBarrier;

       }

       @Override

       public void run() {

           try {

               Thread.sleep(3);

               System.out.println("人员" + Thread.currentThread().getName() + " 到,等待其他人到场");

               cyclicBarrier.await();

           } catch (InterruptedException e) {

               e.printStackTrace();

           } catch (BrokenBarrierException e) {

               e.printStackTrace();

           }

           System.out.println("所有人到齐,可以进行其他活动");

       }

   }

}

人员person1 到,等待其他人到场

人员person3 到,等待其他人到场

人员person4 到,等待其他人到场

人员person2 到,等待其他人到场

人员person5 到,等待其他人到场

所有人到齐,可以进行其他活动

所有人到齐,可以进行其他活动

所有人到齐,可以进行其他活动

所有人到齐,可以进行其他活动

所有人到齐,可以进行其他活动

Semaphore (信号量)

Semaphore 可以控制同时访问的线程个数,通过acquire() 获取一个许可,如果没有就等待,而 release() 释放一个许可。

示例

public class SemaphoreDemo {

   public static void main(String[] args) {

       int N = 5; //购票场景,5个人需要购买票,人工窗口只有2个

       Semaphore semaphore = new Semaphore(2); //人工窗口

       for(int i=1;i<=N;i++)

           new BuyTicketor(i,semaphore).start();

   }

   static class BuyTicketor extends Thread{

       private int num;

       private Semaphore semaphore;

       public BuyTicketor(int num,Semaphore semaphore){

           this.num = num;

           this.semaphore = semaphore;

       }

       @Override

       public void run() {

           try {

               semaphore.acquire();

               System.out.println("购买者"+this.num+"购买票中...");

               Thread.sleep(3);

               System.out.println("购买者"+this.num+"购买完成...");

               semaphore.release();

           } catch (InterruptedException e) {

               e.printStackTrace();

           }

       }

   }

}

输出

购买者1购买票中...

购买者2购买票中...

购买者2购买完成...

购买者1购买完成...

购买者3购买票中...

购买者4购买票中...

购买者4购买完成...

购买者3购买完成...

购买者5购买票中...

购买者5购买完成...


相关文章
|
5月前
|
安全 Java 编译器
揭秘JAVA深渊:那些让你头大的最晦涩知识点,从泛型迷思到并发陷阱,你敢挑战吗?
【8月更文挑战第22天】Java中的难点常隐藏在其高级特性中,如泛型与类型擦除、并发编程中的内存可见性及指令重排,以及反射与动态代理等。这些特性虽强大却也晦涩,要求开发者深入理解JVM运作机制及计算机底层细节。例如,泛型在编译时检查类型以增强安全性,但在运行时因类型擦除而丢失类型信息,可能导致类型安全问题。并发编程中,内存可见性和指令重排对同步机制提出更高要求,不当处理会导致数据不一致。反射与动态代理虽提供运行时行为定制能力,但也增加了复杂度和性能开销。掌握这些知识需深厚的技术底蕴和实践经验。
105 2
|
5月前
|
Java Linux Maven
java依赖冲突解决问题之容器加载依赖jar包如何解决
java依赖冲突解决问题之容器加载依赖jar包如何解决
|
2月前
|
存储 安全 Java
Java多线程编程中的并发容器:深入解析与实战应用####
在本文中,我们将探讨Java多线程编程中的一个核心话题——并发容器。不同于传统单一线程环境下的数据结构,并发容器专为多线程场景设计,确保数据访问的线程安全性和高效性。我们将从基础概念出发,逐步深入到`java.util.concurrent`包下的核心并发容器实现,如`ConcurrentHashMap`、`CopyOnWriteArrayList`以及`BlockingQueue`等,通过实例代码演示其使用方法,并分析它们背后的设计原理与适用场景。无论你是Java并发编程的初学者还是希望深化理解的开发者,本文都将为你提供有价值的见解与实践指导。 --- ####
|
2月前
|
存储 设计模式 分布式计算
Java中的多线程编程:并发与并行的深度解析####
在当今软件开发领域,多线程编程已成为提升应用性能、响应速度及资源利用率的关键手段之一。本文将深入探讨Java平台上的多线程机制,从基础概念到高级应用,全面解析并发与并行编程的核心理念、实现方式及其在实际项目中的应用策略。不同于常规摘要的简洁概述,本文旨在通过详尽的技术剖析,为读者构建一个系统化的多线程知识框架,辅以生动实例,让抽象概念具体化,复杂问题简单化。 ####
|
2月前
|
Java 数据库连接 数据库
如何构建高效稳定的Java数据库连接池,涵盖连接池配置、并发控制和异常处理等方面
本文介绍了如何构建高效稳定的Java数据库连接池,涵盖连接池配置、并发控制和异常处理等方面。通过合理配置初始连接数、最大连接数和空闲连接超时时间,确保系统性能和稳定性。文章还探讨了同步阻塞、异步回调和信号量等并发控制策略,并提供了异常处理的最佳实践。最后,给出了一个简单的连接池示例代码,并推荐使用成熟的连接池框架(如HikariCP、C3P0)以简化开发。
60 2
|
3月前
|
Java
【编程进阶知识】揭秘Java多线程:并发与顺序编程的奥秘
本文介绍了Java多线程编程的基础,通过对比顺序执行和并发执行的方式,展示了如何使用`run`方法和`start`方法来控制线程的执行模式。文章通过具体示例详细解析了两者的异同及应用场景,帮助读者更好地理解和运用多线程技术。
39 1
|
3月前
|
消息中间件 NoSQL Kafka
Flink-10 Flink Java 3分钟上手 Docker容器化部署 JobManager TaskManager Kafka Redis Dockerfile docker-compose
Flink-10 Flink Java 3分钟上手 Docker容器化部署 JobManager TaskManager Kafka Redis Dockerfile docker-compose
78 4
|
3月前
|
Kubernetes Cloud Native 流计算
Flink-12 Flink Java 3分钟上手 Kubernetes云原生下的Flink集群 Rancher Stateful Set yaml详细 扩容缩容部署 Docker容器编排
Flink-12 Flink Java 3分钟上手 Kubernetes云原生下的Flink集群 Rancher Stateful Set yaml详细 扩容缩容部署 Docker容器编排
99 3
|
4月前
|
Java API 容器
JAVA并发编程系列(10)Condition条件队列-并发协作者
本文通过一线大厂面试真题,模拟消费者-生产者的场景,通过简洁的代码演示,帮助读者快速理解并复用。文章还详细解释了Condition与Object.wait()、notify()的区别,并探讨了Condition的核心原理及其实现机制。
|
5月前
|
存储 Java
Java 中 ConcurrentHashMap 的并发级别
【8月更文挑战第22天】
67 5