- 浏览: 57616 次
- 性别:
- 来自: 北京
文章分类
最新评论
Java1.5提供了一个非常高效实用的多线程包:java.util.concurrent, 提供了大量高级工具,可以帮助开发者编写高效、易维护、结构清晰的Java多线程程序。从这篇blog起,我将跟大家一起共同学习这些新的Java多线程构件
1. CountDownLatch
我们先来学习一下JDK1.5 API中关于这个类的详细介绍:
“一个同步辅助类,在完成一组正在其他线程中执行的操作之前,它允许一个或多个线程一直等待。 用给定的计数 初始化
CountDownLatch。由于调用了 countDown() 方法,所以在当前计数到达零之前,await
方法会一直受阻塞。之后,会释放所有等待的线程,await
的所有后续调用都将立即返回。这种现象只出现一次——计数无法被重置。如果需要重置计数,请考虑使用 CyclicBarrier。”
这就是说,CountDownLatch可以用来管理一组相关的线程执行,只需在主线程中调用CountDownLatch
的await方法(一直阻塞),让各个线程调用countDown方法。当所有的线程都只需完countDown了,await也顺利返回,不再阻塞了。
在这样情况下尤其适用:将一个任务分成若干线程执行,等到所有线程执行完,再进行汇总处理。
下面我举一个非常简单的例子。假设我们要打印1-100,最后再输出“Ok“。1-100的打印顺序不要求统一,只需保证“Ok“是在最后出现即可。
解决方案:我们定义一个CountDownLatch,然后开10个线程分别打印(n-1)*10+1至(n-1)*10+10。主线程中调用await
方法等待所有线程的执行完毕,每个线程执行完毕后都调用countDown方法。最后再await返回后打印“Ok”。
具体代码如下(本代码参考了JDK示例代码):
- import java.util.concurrent.CountDownLatch;
- /**
- * 示例:CountDownLatch的使用举例
- * Mail: ken@iamcoding.com
- * @author janeky
- */
- public class TestCountDownLatch {
- private static final int N = 10 ;
- public static void main(String[] args) throws InterruptedException {
- CountDownLatch doneSignal = new CountDownLatch(N);
- CountDownLatch startSignal = new CountDownLatch( 1 ); //开始执行信号
- for ( int i = 1 ; i <= N; i++) {
- new Thread( new Worker(i, doneSignal, startSignal)).start(); //线程启动了
- }
- System.out.println( "begin------------" );
- startSignal.countDown(); //开始执行啦
- doneSignal.await(); //等待所有的线程执行完毕
- System.out.println( "Ok" );
- }
- static class Worker implements Runnable {
- private final CountDownLatch doneSignal;
- private final CountDownLatch startSignal;
- private int beginIndex;
- Worker( int beginIndex, CountDownLatch doneSignal,
- CountDownLatch startSignal) {
- this .startSignal = startSignal;
- this .beginIndex = beginIndex;
- this .doneSignal = doneSignal;
- }
- public void run() {
- try {
- startSignal.await(); //等待开始执行信号的发布
- beginIndex = (beginIndex - 1 ) * 10 + 1 ;
- for ( int i = beginIndex; i <= beginIndex + 10 ; i++) {
- System.out.println(i);
- }
- } catch (InterruptedException e) {
- e.printStackTrace();
- } finally {
- doneSignal.countDown();
- }
- }
- }
- }
总结:CounDownLatch对于管理一组相关线程非常有用。上述示例代码中就形象地描述了两种使用情况。第一种是计算器为1,代表了两种状态,开
关。第二种是计数器为N,代表等待N个操作完成。今后我们在编写多线程程序时,可以使用这个构件来管理一组独立线程的执行。
2. CyclicBarrier
我们先来学习一下JDK1.5 API中关于这个类的详细介绍:
“一个同步辅助类,它允许一组线程互相等待,直到到达某个公共屏障点 (common barrier
point)。在涉及一组固定大小的线程的程序中,这些线程必须不时地互相等待,此时 CyclicBarrier 很有用。因为该 barrier
在释放等待线程后可以重用,所以称它为循环 的 barrier。
CyclicBarrier 支持一个可选的 Runnable 命令,在一组线程中的最后一个线程到达之后(但在释放所有线程之前),该命令只在每个屏障点运行一次。若在继续所有参与线程之前更新共享状态,此屏障操作 很有用。
我们在学习CountDownLatch的时候就提到了CyclicBarrier。两者究竟有什么联系呢?引用[JCIP]中的描述“The
key difference is that with a barrier, all the threads must come
together at a barrier point at the same time in order to proceed.
Latches are for waiting for events; barriers are for waiting
for other
threads。CyclicBarrier等待所有的线程一起完成后再执行某个动作。这个功能CountDownLatch也同样可以实现。但是
CountDownLatch更多时候是在等待某个事件的发生。在CyclicBarrier中,所有的线程调用await方法,等待其他线程都执行完。
举一个很简单的例子,今天晚上我们哥们4个去Happy。就互相通知了一下:晚上八点准时到xx酒吧门前集合,不见不散!。有个哥们住的近,早早就到了。有的事务繁忙,刚好踩点到了。无论怎样,先来的都不能独自行动,只能等待所有人
代码如下(参考了网上给的一些教程)
- import java.util.Random;
- import java.util.concurrent.BrokenBarrierException;
- import java.util.concurrent.CyclicBarrier;
- import java.util.concurrent.ExecutorService;
- import java.util.concurrent.Executors;
- public class TestCyclicBarrier {
- public static void main(String[] args) {
- ExecutorService exec = Executors.newCachedThreadPool();
- final Random random= new Random();
- final CyclicBarrier barrier= new CyclicBarrier( 4 , new Runnable(){
- @Override
- public void run() {
- System.out.println( "大家都到齐了,开始happy去" );
- }});
- for ( int i= 0 ;i< 4 ;i++){
- exec.execute( new Runnable(){
- @Override
- public void run() {
- try {
- Thread.sleep(random.nextInt( 1000 ));
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
- System.out.println(Thread.currentThread().getName()+ "到了,其他哥们呢" );
- try {
- barrier.await(); //等待其他哥们
- } catch (InterruptedException e) {
- e.printStackTrace();
- } catch (BrokenBarrierException e) {
- e.printStackTrace();
- }
- }});
- }
- exec.shutdown();
- }
- }
关于await方法要特别注意一下,它有可能在阻塞的过程中由于某些原因被中断
总结:CyclicBarrier就是一个栅栏,等待所有线程到达后再执行相关的操作。barrier 在释放等待线程后可以重用。
发表评论
-
Lock和读写锁ReadWriteLock和缓存实例
2012-10-22 16:54 11301:lock和synchronized对比 [ja ... -
ReentrantLock与synchronized的区别
2012-10-22 13:33 520ReentrantLock 的lock机制有2种,忽略中断 ... -
java的concurrent/java.util.concurrent.locks用法详解
2012-10-19 15:49 721java.util.concurrent.locks - ... -
java.util.concurrent详解(四) BlockingQueue
2012-10-19 10:00 7597.BlockingQueue “支持两个附加操作 ... -
java.util.concurrent详解(三)ScheduledThreadPoolExecutor
2012-10-18 14:32 8306. ScheduledThreadPoolExecutor ... -
java.util.concurrent详解(二)Semaphore/FutureTask/Exchanger
2012-10-18 14:31 968------------------------------- ... -
浅析Java中CountDownLatch用法
2012-10-16 18:09 8CountDownLatch如其所写,是一个倒计数的锁存器,当 ... -
Java轻量级锁原理详解(Lightweight Locking)
2012-10-11 18:06 797大家知道,Java的多线程 ... -
深入理解DCL(双检锁)的安全性
2012-10-11 18:07 948对于双检锁,其实有多种不同的用法,有很多种用法是无论如何不会出 ... -
Java偏向锁实现原理(Biased Locking)
2012-10-11 18:08 745阅读本文的读者,需要对Java轻量级锁有一定的了解,知道loc ... -
多线程下race condition问题
2012-10-11 18:07 1413这个问题的讨论来自内部的一个关于“多线程环境下使用Hashma ...
相关推荐
1. java.util.concurrent - Java 并发工具包 2. 阻塞队列 BlockingQueue 3. 数组阻塞队列 ArrayBlockingQueue 4. 延迟队列 DelayQueue 5. 链阻塞队列 LinkedBlockingQueue 6. 具有优先级的阻塞队列 ...
`java.util.concurrent`包是Java提供的一个强大的多线程工具库,其中包含了许多类和接口,如`CountDownLatch`和`CyclicBarrier`,它们为程序员提供了更高级别的同步和协调机制。这篇文档将详细解析这两个工具类的...
"java.util.concurrent.ExecutionException: java.lang.OutOfMemoryError" 是一个典型的错误提示,它表明在并发执行过程中遇到了内存不足的问题。下面我们将深入探讨这个问题的原因、影响以及如何解决。 内存溢出...
### Java并发工具包 `java.util.concurrent` 知识点详解 #### 一、引言 随着多核处理器的普及和应用程序复杂度的增加,多线程编程成为了现代软件开发不可或缺的一部分。为了简化并发编程的复杂性,Java 5 引入了 `...
Java.util.concurrent是Java 5.0引入的一个重要包,它为多线程编程提供了一组高级并发工具。这个包的设计者是Doug Lea,它的出现是JSR-166的一部分,也被称作Tiger更新。Java.util.concurrent的引入是为了解决传统...
java.util.concurrent总体概览图。 收取资源分3分。需要的同学可以下载一下。 java.util.concurrent主要包括5个部分executor,colletions,locks,atomic,tools。 该图详细的列举了并发包下面的结构,包含所有接口和...
Java.util.concurrent(JUC)是Java平台中的一个核心包,专门用于处理多线程并发问题。这个包包含了大量的工具类和接口,极大地简化了并发编程的复杂性,提高了程序的性能和可伸缩性。本测试源文件主要是针对JUC并发...
JDK1.5中的线程池(java.util.concurrent.ThreadPoolExecutor)使用是Java多线程编程中的一种重要概念。随着多线程编程的普及,线程池的使用变得越来越频繁。Java中的线程池是通过ThreadPoolExecutor类实现的。 一、...
### Java.util.concurrent.Synchronizer框架详解 #### 一、引言与背景 随着Java技术的发展,多线程编程成为了一项重要的技术需求。为了更好地支持并发编程,Java平台在J2SE 1.5版本中引入了`java.util.concurrent`...
从继承关系上来看,`java.sql.Date` 是从 `java.util.Date` 继承而来,这意味着 `java.sql.Date` 实际上就是 `java.util.Date` 的一个子类。这种设计使得 `java.sql.Date` 自然地拥有 `java.util.Date` 的所有特性,...
java.util.concurrent - Java 并发工具包 2. 阻塞队列 BlockingQueue 3. 数组阻塞队列 ArrayBlockingQueue 4. 延迟队列 DelayQueue 5. 链阻塞队列 LinkedBlockingQueue 6. 具有优先级的阻塞队列 ...
Java.util.ConcurrentModificationException 异常问题详解 ConcurrentModificationException 异常是 Java 中一个常见的异常,它发生在 Iterator 遍历集合时,集合同时被修改引起的异常。在 Java 中,集合类如 ...
本文将通过一系列的示例来详细介绍如何利用 Java 中的 `java.util.zip` 包进行数据的压缩与解压操作。此外,还将探讨这一技术在网络传输中的应用。 #### 数据压缩算法简介 在深入讨论具体实现之前,我们先简要了解...
#### 一、Java.util.Date与Java.sql.Date的基本概念 在Java编程语言中,处理日期和时间时经常使用到`java.util.Date`和`java.sql.Date`这两个类。它们虽然名字相似,但在实际应用中有很大的区别。 - **`java.util....
### Java.util.logging.Logger 使用详解 #### 一、创建Logger对象 在Java中,`java.util.logging.Logger` 是标准的日志框架之一,它提供了基础的日志记录功能。为了使用这一功能,首先需要获得 `java.util.logging...
一个高性能的Java线程库,该库是 JDK 1.5 中的 java.util.concurrent 包的补充,可用于基于并发消息机制的应用。该类库不提供远程的消息功能,其设计的宗旨是实现一个内存中的消息传递机制. 主要特点有: * All ...
在Java编程语言中,`java.util.InputMismatchException`是一个常见的运行时异常,它通常发生在尝试从数据源(如控制台、文件或数据库)读取数据时,遇到的数据类型与预期的不匹配。在这个特定的场景中,问题出在主线...
多线程编程是 Java 语言中的一大特点,java.util.concurrent 多线程框架则是该特点的体现。该框架提供了多种机制,包括线程池、并发集合、同步器、lock 等,以便开发者更方便地编写高效、可靠的多线程程序。 在 ...