`

透过源码和实例学习CountDownLatch类

 
阅读更多

最近在做一个程序的 时候遇到了这样的情况,在 主类中需要执行一些操作,同时主类会启动几个线程,在这些线程执行前和执行后都会一系列的操作。因此就用到了CountDownLatch类。CountDownLatch是一个同步辅助类,在完成一组正在其他线程中执行的操作之前,它允许一个或多个线程一直等待。

 

下边是一个例子:

import mulithread.CountDownLatch;

class Driver {  
	public static void main(String []args) throws InterruptedException {
		CountDownLatch startSignal = new CountDownLatch(1,"startsignal");
		CountDownLatch doneSignal = new CountDownLatch(2,"endsignal");
		
		for (int i = 0; i < 2; ++i)
			new Thread(new Worker(startSignal, doneSignal,i)).start();// create and start threads
		Thread.sleep(3000);
		System.out.println("don't let run yet"); //don't let run yet
		startSignal.countDown(); //let all threads proceed
		doneSignal.await(); // wait for all to finish
		System.out.println("wait for all to finish");
	}
}

 

import mulithread.CountDownLatch;


class Worker implements Runnable {
	private final CountDownLatch startSignal;
	private final CountDownLatch doneSignal;
	private int threadid;
	Worker(CountDownLatch startSignal, CountDownLatch doneSignal,int threadid) {
		this.startSignal = startSignal;
		this.doneSignal = doneSignal;
		this.threadid=threadid;
	} 

	public void run() {
		try {
			System.out.println(threadid+"before process");
			startSignal.await();//所有调用 await 的线程都一直在入口处等待
			doWork();
			doneSignal.countDown();
			System.out.println(threadid+"after process");
		} catch (InterruptedException ex) {
		
		}  
	}

	void doWork() {
		try {
			Thread.sleep(20);
		} catch (InterruptedException e) {
			e.printStackTrace();
		}
		System.out.println(threadid+"runing----------");
	}
}

 

import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.AbstractQueuedSynchronizer;

public class CountDownLatch {
	
    private static final class Sync extends AbstractQueuedSynchronizer {
        private static final long serialVersionUID = 4982264981922014374L;

        Sync(int count) {
            setState(count);
        }

        int getCount() {
            return getState();
        }

        public int tryAcquireShared(int acquires) {
            return getState() == 0? 1 : -1;
        }

        public boolean tryReleaseShared(int releases) {
            // Decrement count; signal when transition to zero
            for (;;) {
                int c = getState();
                if (c == 0)
                    return false;
                int nextc = c-1;
                if (compareAndSetState(c, nextc))
                    return nextc == 0;
            }
        }
    }
    private String name;
    private final Sync sync;

    public CountDownLatch(int count,String name) {
    	this.name=name;
        if (count < 0) throw new IllegalArgumentException("count < 0");
        this.sync = new Sync(count);
    }

    public void await() throws InterruptedException {
        sync.acquireSharedInterruptibly(1);
    	System.out.println(name+" await invoked!"+getCount());
    }
    public boolean await(long timeout, TimeUnit unit)
        throws InterruptedException {
        return sync.tryAcquireSharedNanos(1, unit.toNanos(timeout));
    }

    public void countDown() {
        sync.releaseShared(1);
        System.out.println(name+" count down invoked!"+getCount());
    }
    public long getCount() {
        return sync.getCount();
    }

    public String toString() {
        return super.toString() + "[Count = " + sync.getCount() + "]";
    }
}

 

注意,有三个类,其中Driver是主类,Worker类是要工作时候用到的线程,CountDownLatch类值从java源代码抠出来的,只是做了一些状态输出工作。在这个例子中,有两个工作线程,所有的线程在工作之前(dowork方法完成工作任务)startSignal的await()方法会一直阻塞,直到主类中startSignal的countDown()执行之后才会执行。 同样的,在所有工作完成之前,主类中doneSignal的await()方法会一直阻塞,一直到doneSignal的所有线程的 countdown方法会被调用,使得计数器的值递减,从而后边的代码才会执行。这个例子其实简单而言描述这样一个场景:主类中如果有几个线程,但是这些线程又必须在主类中的一些操作完成之前才可以执行,那么就可以把代码放在主类的startSignal的countdown()方法之上,同样,如果想要在所有线程执行之后,做一些工作,那么就可以把代码放在doneSignal的await()方法之后执行。至于CountDownLatch 中的参数,其实就是计数器的值,看你的具体需求情况啦!

分享到:
评论

相关推荐

    CountDownLatch和CyclicBarrier用法实例大全

    在Java并发编程中,CountDownLatch和CyclicBarrier是两种非常重要的同步工具类,它们用于协调多个线程间的协作。这两个工具都是在`java.util.concurrent`包下,是Java并发库的重要组成部分。 **CountDownLatch** ...

    140个Java源码实例

    "140个Java源码实例"是一份珍贵的学习资源,它包含了多个实际编程场景下的代码示例,旨在帮助初学者更好地理解和掌握Java编程。 在学习这140个Java源码实例时,我们可以按照以下主要知识点进行分类和深入学习: 1....

    利用 CountDownLatch 类实现线程同步

    Java 提供了多种工具来实现这样的同步机制,其中之一便是 `CountDownLatch` 类。`CountDownLatch` 是一个计数器,可以用于协调多个线程间的活动,等待所有线程完成各自的任务后,主线程或其他线程才能继续执行。 ...

    140个Java源码实例.zip

    源码实例可能包含这些基本元素,帮助学习者建立扎实的语法基础。 2. **类与对象**:Java是面向对象的语言,因此,类的定义、对象的创建和方法的使用是核心内容。实例中可能会有各种类的设计,例如封装、继承和多态...

    多线程countDownLatch方法介绍

    在Java多线程编程中,CountDownLatch是一个非常重要的同步工具类,它可以帮助我们协调多个线程之间的交互。本文将深入探讨CountDownLatch的工作原理、使用场景以及相关源码分析。 CountDownLatch是一个计数器,初始...

    基于Java的实例源码-多线程反射泛型及正则表达式学习笔记和源码.zip

    总的来说,这个压缩包中的源码和学习笔记是深入理解并熟练运用Java多线程、反射、泛型和正则表达式的宝贵资源。通过研究这些实例,开发者不仅可以巩固理论知识,还能提升实际编程技巧,从而在日常工作中编写出更加...

    JAVA实例精通源代码

    6. **多线程**:Java提供了丰富的多线程支持,源码中会包含Thread、Runnable接口的使用,以及同步机制如synchronized关键字、wait/notify机制和并发工具类(如Semaphore、CountDownLatch等)的实例。 7. **网络编程...

    CountDownLatch学习用法

    CountDownLatch是Java并发编程中一个重要的工具类,它属于java.util.concurrent包下的一个同步辅助类。这个类的设计目的是允许一个线程等待其他多个线程完成操作,然后再继续执行。CountDownLatch通常用于多线程协作...

    java并发编程专题(八)----(JUC)实例讲解CountDownLatch

    本篇文章主要介绍了Java中的CountDownLatch类,作为Java并发编程专题(八)的一部分,通过实例讲解帮助读者理解和学习。下面是文章中的知识点总结: 一、CountDownLatch简介 CountDownLatch是一个同步辅助类,允许...

    java并发编程中CountDownLatch和CyclicBarrier的使用借鉴.pdf

    java并发编程中CountDownLatch和CyclicBarrier的使用借鉴 java并发编程中CountDownLatch和CyclicBarrier是两个非常重要的线程控制和调度工具,经常被用于解决多线程程序设计中的线程等待问题。本文将对...

    android实例源码

    "android实例源码"这个压缩包提供了一些简单但实用的Android项目源代码,可以帮助开发者更好地理解和掌握Android编程的基本概念、组件以及设计模式。以下是这些源码实例中可能包含的一些关键知识点: 1. **Activity...

    Java中的CountDownLatch类最全讲义

    1.2 CountDownLatch概述 CountDownLatch的基本用法 2.1 创建CountDownLatch对象 2.2 await()方法 2.3 countDown()方法 实现多线程任务的同步 3.1 场景介绍 3.2 使用CountDownLatch实现同步 主线程等待多个线程完成 ...

    android核心技术与实例详解口袋微博项目源码

    在"KDWB"这个压缩包中,可能包含了上述各个技术的实现细节,通过阅读和分析这些源码,开发者不仅可以学习到具体的实现方式,还能加深对Android系统和开发流程的理解。对于希望深入学习Android开发的工程师来说,这样...

    收集的Java高级编程实例源码

    这个压缩包中的“Java高级编程实例源码”集合,为学习者提供了丰富的实践素材,帮助深入理解Java的核心特性和高级用法。下面我们将逐一探讨这些知识点。 1. **多线程编程**:Java的并发处理能力是其一大亮点,通过`...

    mybaits 多线程 实现数据批量插入 (运用CountDownLatch实现闭锁)

    在Java并发编程中,`CountDownLatch`是一个同步辅助类,它允许多个线程等待其他线程完成操作。在批量插入数据的场景下,可以创建一个CountDownLatch对象,初始化为线程的数量,每个线程处理完自己的数据后调用`...

    CountDownLatch源码解析之await()

    CountDownLatch源码解析之await() CountDownLatch是Java并发编程中常用的同步工具,通过await()方法可以让当前线程处于阻塞状态,直到锁存器计数为零(或者线程中断)。下面我们将详细解析CountDownLatch源码之...

    Java中CountDownLatch进行多线程同步详解及实例代码

    下面是对CountDownLatch的详细解释和实例代码。 CountDownLatch的实现原理简介: CountDownLatch内部维护一个计数器,计数器的值为待完成的任务数N。需要等待这N个任务完成的线程调用CountDownLatch的await()方法使...

    《Java多线程编程实例》随书源码

    通过《Java多线程编程实例》随书源码,读者可以深入了解并实践这些概念,例如创建和管理线程池、实现线程安全的数据结构、解决死锁问题以及优化多线程应用等。书中的实例涵盖了多线程编程的各个重要方面,对于提升...

    countdownlatch-example-sourcecode.zip

    《CountDownLatch实战解析与源码探索》 CountDownLatch是Java并发编程中一个非常重要的同步工具类,它在多线程协作...同时,掌握好CountDownLatch和其他并发工具类,能够帮助我们编写出更加高效、稳定的多线程程序。

    28个java常用的工具类源码

    18. **CountDownLatch和CyclicBarrier**:同步原语,用于多线程间的协调。 19. **Semaphore**:信号量,用于控制并发访问的资源数量。 20. **Future和ExecutorService**:异步编程和任务调度的重要组件。 21. **...

Global site tag (gtag.js) - Google Analytics