`
脑壳冒星星
  • 浏览: 15616 次
文章分类
社区版块
存档分类
最新评论

java CountDownLatch CyclicBarrier

 
阅读更多

CountDownLatch: 一个或者是一部分线程 ,等待另外一部线程都完成了,再继续执行
       CyclicBarrier: 所有线程互相等待完成。

 

       CyclicBarrier ,一个同步辅助类,它允许一组线程互相等待,直到到达某个公共屏障点 (common barrier point)。在涉及一组固定大小的线程的程序中,这些线程必须不时地互相等待,此时 CyclicBarrier 很有用。因为该 barrier 在释放等待线程后可以重用,所以称它为循环 的 barrier。

CyclicBarrier可以多次重复使用

 

CountDownLatch,一个同步辅助类,在完成一组正在其他线程中执行的操作之前,它允许一个或多个线程一直等待。用给定的计数 初始化 CountDownLatch。由于调用了 countDown() 方法,所以在当前计数到达零之前,await 方法会一直受阻塞。之后,会释放所有等待的线程,await 的所有后续调用都将立即返回。这种现象只出现一次——计数无法被重置。 一个线程(或者多个), 等待另外N个线程完成某个事情之后才能执行,只能使用一次。

 

CountDownLatch主要方法
public CountDownLatch(int count);
public void countDown();
public void await() throws InterruptedException

构造方法参数指定了计数的次数
countDown方法,当前线程调用此方法,则计数减一
awaint方法,调用此方法会一直阻塞当前线程,直到计时器的值为0

 

Java代码

    public class CountDownLatchDemo {  
        final static SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");  
        public static void main(String[] args) throws InterruptedException {  
            CountDownLatch latch=new CountDownLatch(2);//两个工人的协作  
            Worker worker1=new Worker("zhang san", 5000, latch);  
            Worker worker2=new Worker("li si", 8000, latch);  
            worker1.start();//  
            worker2.start();//  
            latch.await();//等待所有工人完成工作  
            System.out.println("all work done at "+sdf.format(new Date()));  
        }  
          
        static class Worker extends Thread {  
            String workerName;   
            int workTime;  
            CountDownLatch latch;  
            public Worker(String workerName ,int workTime ,CountDownLatch latch){  
                 this.workerName=workerName;  
                 this.workTime=workTime;  
                 this.latch=latch;  
            }  
            public void run() {  
                System.out.println("Worker "+workerName+" do work begin at "+sdf.format(new Date()));  
                doWork();//工作了  
                System.out.println("Worker "+workerName+" do work complete at "+sdf.format(new Date()));  
            }  
              
            private void doWork() {  
                try {  
                    Thread.sleep(workTime);  
                } catch (InterruptedException e) {  
                    e.printStackTrace();  
                }  finally {
                    latch.countDown();
                }  
             }
         }

输出:

Worker zhang san do work begin at 2014-10-17 11:05:11
Worker li si do work begin at 2014-10-17 11:05:11
Worker zhang san do work complete at 2014-10-17 11:05:16
Worker li si do work complete at 2014-10-17 11:05:19
all work done at 2014-10-17 11:05:19

 

    package com.study.thread;  
      
    import java.util.Random;  
    import java.util.concurrent.CyclicBarrier;  
    import java.util.concurrent.ExecutorService;  
    import java.util.concurrent.Executors;  
    import java.util.concurrent.TimeUnit;  
      
    /** 
     * 用java模拟4X100接力赛 
     */  
    class Player implements Runnable{  
        private String name;  
        private CyclicBarrier barrier;  
        private Player next;//下一棒  
        private int time;//用时  
        private boolean run;//第一棒  
        public Player(String name, CyclicBarrier barrier, boolean run) {  
            super();  
            this.name = name;  
            this.barrier = barrier;  
            this.run = run;  
        }  
        @Override  
        public void run() {  
            try {  
                synchronized (this) {  
                    while(!run){//等待队员  
                        wait();  
                    }  
                }  
                Random r = new Random();  
                TimeUnit.MILLISECONDS.sleep(r.nextInt(2000));  
                next(next,11 + r.nextInt(2));  
            } catch (InterruptedException e) {  
                e.printStackTrace();  
            }  
              
        }  
        private void next(Player next, int time) {  
            System.out.println(name + " 用时:" + time + ",交接棒");  
            if(next != null){  
                next.setTime(this.time + time);  
                synchronized (next) {  
                    next.setRun(true);  
                    next.notify();  
                }  
            }else{  
                System.out.println("跑完,总用时:" + (this.time + time));  
            }  
        }  
        public void setTime(int time) {  
            this.time = time;  
        }  
        public int getTime(){  
            return this.time;  
        }  
        public void setNext(Player next) {  
            this.next = next;  
        }  
        public void setRun(boolean run) {  
            this.run = run;  
        }  
          
    }  
    public class RelayRace {  
          
        public static void main(String[] args) throws InterruptedException {  
            final Player[] players = new Player[4];  
            ExecutorService exec = Executors.newCachedThreadPool();  
            CyclicBarrier barrier = new CyclicBarrier(4, new Runnable() {  
                  
                @Override  
                public void run() {  
                    System.out.println("结束,总用时:" + players[3].getTime());  
                }  
            });  
            for(int i = 0; i < 4; i++){  
                players[i] = new Player("队员" + ( i + 1), barrier, i == 0);  
            }  
            for(int i = 0; i < 4; i++){  
                if( i < 3){  
                    players[i].setNext(players[i + 1]);  
                    exec.execute(players[i]);  
                }else{  
                    exec.execute(players[3]);  
                    break;  
                }  
            }  
            /*TimeUnit.SECONDS.sleep(3); 
             * CyclicBarrier 可以重用 
            for(int i = 0; i < 4; i++){ 
                if( i < 3){ 
                    players[i].setNext(players[i + 1]); 
                    exec.execute(players[i]); 
                }else{ 
                    exec.execute(players[3]); 
                    break; 
                } 
            }*/  
        }  
      
    }  

 输出:

队员1 用时:11,交接棒

队员2 用时:11,交接棒

队员3 用时:11,交接棒

队员4 用时:12,交接棒

跑完,总用时:45

分享到:
评论

相关推荐

    CountDownLatch和CyclicBarrier用法实例大全

    《java并发编程》中CountDownLatch和CyclicBarrier用法实例大全,几乎包含了所有重要的用法

    详解java CountDownLatch和CyclicBarrier在内部实现和场景上的区别

    主要介绍了详解java CountDownLatch和CyclicBarrier在内部实现和场景上的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

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

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

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

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

    CountDownLatch 和 CyclicBarrier 的运用(含AQS详解)

    CountDownLatch 和 CyclicBarrier 为线程同步的辅助工具,通过它可以做到使一条线程一直阻塞等待,直到其他线程完成其所处理的任务。

    Java并发编程一CountDownLatch、CyclicBarrier、Semaphore初使用

    Java并发编程一CountDownLatch、CyclicBarrier、Semaphore初使用 CountDownLatch、CyclicBarrier、Semaphore这些线程协作工具类是基于AQS的,看完这篇博客后可以去看下面这篇博客,了解它们是如何实现的。 Java并发...

    CountDownLatch、CyclicBarrier、Semaphore.md

    java 高并发应用场景

    Java中CyclicBarrier的用法分析

    CyclicBarrier和CountDownLatch一样,都是关于线程的计数器。用法略有不同,测试代码如下:

    Java并发编程:CountDownLatch与CyclicBarrier和Semaphore的实例详解

    主要介绍了Java并发编程:CountDownLatch与CyclicBarrier和Semaphore的实例详解的相关资料,需要的朋友可以参考下

    CyclicBarrier用法.docx

    CyclicBarrier是Java中的一个同步工具类,用于协调多个线程之间的同步。它的作用类似于CountDownLatch,可以让一个或多个线程等待其他线程执行完毕后再继续执行。 CyclicBarrier可以被看作是一个屏障,当所有线程都...

    Java中的CyclicBarrier类最全讲义

    目录: 简介 1.1 并发编程与线程协作 1.2 CyclicBarrier概述 CyclicBarrier的基本用法 2.1 创建CyclicBarrier对象 2.2 await()方法 控制线程协作数量 ...CyclicBarrier与CountDownLatch的对比 最佳实践与注意事项

    Java并发包讲解

    java并发包讲解 可以找我要代码,qq 3341386488 ## 线程安全-并发容器JUC--原理以及分析 1.arrayList --copyonWriteArraylist 优缺点 2.HashSet,TreeSet -- CopyONWriteArraySet,ConcurrentSkipListSet 3....

    Java中的CountDownLatch类最全讲义

    目录: 简介 1.1 并发编程与线程同步 1.2 CountDownLatch概述 CountDownLatch的基本用法 2.1 创建CountDownLatch对象 2.2 await()方法 ...CountDownLatch与CyclicBarrier的对比 最佳实践与注意事项

    Java线程面试题大全.docx

    1. 什么是线程? 2. 线程和进程有什么区别? 3. 如何在 Java 中实现...7. Java 中 CyclicBarrier 和 CountDownLatch 有什么不同? 8. Java 内存模型是什么? 9. Java 中的 volatile 变量是什么? ......... ........

    多线程下的并发包:这些面试必问的,你都知道吗?

    文章目录...Java为很多业务场景提供了性能优异,且线程安全的并发包,程序员可以选择使用! ConcurrentHashMap 为什么要使用ConcurrentHashMap: HashMap线程

    Java进阶教程,面试大全

    Java进阶教程,面试大全1,可参考以下问题: Semaphore-信号灯机制。 synchronized在静态方法和普通方法的区别。 怎么实现所有线程在等待某个事件的发生...CountDownLatch和CyclicBarrier的用法,以及相互之间的差别。

    Java进阶教程,面试大全,包罗万象

    Java进阶教程,面试大全1,可参考以下问题: Semaphore-信号灯机制。 synchronized在静态方法和普通方法的区别。 怎么实现所有线程在等待某个事件的发生...CountDownLatch和CyclicBarrier的用法,以及相互之间的差别。

    【Java入门知识图谱】帮助Java初学者成长

    【对线面试官】CountDownLatch和CyclicBarrier 【对线面试官】为什么需要Java内存模型? 【对线面试官】深入浅出Java内存模型 Java虚拟机 【对线面试官】Java从编译到执行,发生了什么? 【对线面试官】双亲委派机制...

    Java并发编程基础.pdf

    Java并发编程基础主要包括以下几个核心方面: ...并发工具类:掌握Java并发包java.util.concurrent中提供的各种工具类,如CountDownLatch、CyclicBarrier、Semaphore等,它们简化了并发编程的复杂性。

    Java并发编程学习笔记

    7、并发工具类CountDownLatch 、CyclicBarrier和Semaphore底层实现原理 8、线程池原理和如何使用线程池 9、ThreadLocal 为什么会内存泄漏 10、Volatile底层实现原理 11、AQS源码分析 12、CAS原理分析和使用场景 13、...

Global site tag (gtag.js) - Google Analytics