java中的线程(4):常用同步类 CountDownLatch、CyclicBarrier和Semaphore
转自:
http://www.cnblogs.com/dolphin0520/p/3920397.html
1.简介
- CountDownLatch和CyclicBarrier都能够实现线程之间的等待,只不过它们侧重点不同。
- CountDownLatch一般用于某个线程A等待若干个其他线程执行完任务之后,它才执行
- CyclicBarrier 回环栅栏,一般用于一组线程互相等待至某个状态,然后这一组线程再同时开始执行后续任务
- CountDownLatch 在使用完后不可重用
- CyclicBarrier 使用完后可重用
- Semaphore 信号量,一般用于控制对某组资源的访问权限。
2.CountDownLatch示例
void testCountDownLatch() { final CountDownLatch latch = ); new Thread() { public void run() { try { Log.d(TAG, "子线程" + Thread.currentThread().getName() + "正在执行"); Thread.sleep(); Log.d(TAG, "子线程" + Thread.currentThread().getName() + "执行完毕"); latch.countDown(); } catch (InterruptedException e) { e.printStackTrace(); } } ; }.start(); new Thread() { public void run() { try { Log.d(TAG, "子线程" + Thread.currentThread().getName() + "正在执行"); Thread.sleep(); Log.d(TAG, "子线程" + Thread.currentThread().getName() + "执行完毕"); latch.countDown(); } catch (InterruptedException e) { e.printStackTrace(); } } ; }.start(); try { Log.d(TAG, "等待2个子线程执行完毕..."); latch.await(); Log.d(TAG, "2个子线程已经执行完毕"); Log.d(TAG, "继续执行主线程"); } catch (InterruptedException e) { e.printStackTrace(); } }
3.CyclicBarrier示例
static class Writer extends Thread { private CyclicBarrier cyclicBarrier; public Writer(CyclicBarrier cyclicBarrier) { this.cyclicBarrier = cyclicBarrier; } @Override public void run() { Log.d(TAG, "线程" + Thread.currentThread().getName() + "正在写入数据..."); try { Thread.sleep(); //以睡眠来模拟写入数据操作 Log.d(TAG, "线程" + Thread.currentThread().getName() + "写入数据完毕,等待其他线程写入完毕"); cyclicBarrier.await(); } catch (InterruptedException e) { e.printStackTrace(); } catch (BrokenBarrierException e) { e.printStackTrace(); } Log.d(TAG, "所有线程写入完毕,继续处理其他任务..."); } } //用来阻塞当前线程,直至所有线程都到达barrier状态再同时执行后续任务; void testCyclicBarrier1(){ ; CyclicBarrier barrier = new CyclicBarrier(N); ; i < N; i++) { new Writer(barrier).start(); } } //如果说想在所有线程写入操作完之后,进行额外的其他操作可以为CyclicBarrier提供Runnable参数 void testCyclicBarrier2(){ ; CyclicBarrier barrier = new CyclicBarrier(N, new Runnable() { @Override public void run() { Log.d(TAG, "当前线程" + Thread.currentThread().getName()); } }); ; i < N; i++){ new Writer(barrier).start(); } } //CyclicBarrier是可以重用的 void testCyclicBarrier3(){ ; CyclicBarrier barrier = new CyclicBarrier(N); ; i < N; i++) { new Writer(barrier).start(); } try { Thread.sleep(); } catch (InterruptedException e) { e.printStackTrace(); } Log.d(TAG, "CyclicBarrier重用"); ; i < N; i++) { new Writer(barrier).start(); } } //故意让最后一个线程启动延迟,因为在前面三个线程都达到barrier之后, //等待了指定的时间发现第四个线程还没有达到barrier,就抛出异常并继续执行后面的任务。 void testCyclicBarrier4(){ ; CyclicBarrier barrier = new CyclicBarrier(N); ; i < N; i++) { ) new Writer2(barrier).start(); else { try { Thread.sleep(); } catch (InterruptedException e) { e.printStackTrace(); } new Writer(barrier).start(); } } } static class Writer2 extends Thread{ private CyclicBarrier cyclicBarrier; public Writer2(CyclicBarrier cyclicBarrier) { this.cyclicBarrier = cyclicBarrier; } @Override public void run() { Log.d(TAG, "线程"+Thread.currentThread().getName()+"正在写入数据..."); try { Thread.sleep(); //以睡眠来模拟写入数据操作 Log.d(TAG, "线程"+Thread.currentThread().getName()+"写入数据完毕,等待其他线程写入完毕"); try { cyclicBarrier., TimeUnit.MILLISECONDS);//等待了指定的时间(2ms)发现第四个线程还没有达到barrier,就抛出异常并继续执行后面的任务 } catch (TimeoutException e) { // TODO Auto-generated catch block e.printStackTrace(); } } catch (InterruptedException e) { e.printStackTrace(); }catch(BrokenBarrierException e){ e.printStackTrace(); } Log.d(TAG, Thread.currentThread().getName()+"所有线程写入完毕,继续处理其他任务..."); } }
4.Semaphore示例
void testSemaphore() { ; //工人数 Semaphore semaphore = ); //机器数目 ; i < N; i++) new Worker(i, semaphore).start(); } static class Worker extends Thread { private int num; private Semaphore semaphore; public Worker(int num, Semaphore semaphore) { this.num = num; this.semaphore = semaphore; } @Override public void run() { try { semaphore.acquire(); Log.d(TAG,"工人" + this.num + "占用一个机器在生产..."); Thread.sleep(); Log.d(TAG,"工人" + this.num + "释放出机器"); semaphore.release(); } catch (InterruptedException e) { e.printStackTrace(); } } }
java中的线程(4):常用同步类 CountDownLatch、CyclicBarrier和Semaphore的更多相关文章
- Java中的4个并发工具类 CountDownLatch CyclicBarrier Semaphore Exchanger
在 java.util.concurrent 包中提供了 4 个有用的并发工具类 CountDownLatch 允许一个或多个线程等待其他线程完成操作,课题点 Thread 类的 join() 方法 ...
- Java中Collection、Map常用实现类研究分析
接口/实现类 描述 key是否可为null 为null是否报错 key是否重复 key重复是否报错 key是否和添加一致 是否线程安全 List 一组元素的集合 ArrayList 基于数组存储,读取 ...
- Java中创建线程的三种方式以及区别
在java中如果要创建线程的话,一般有3种方法: 继承Thread类: 实现Runnable接口: 使用Callable和Future创建线程. 1. 继承Thread类 继承Thread类的话,必须 ...
- 浅谈利用同步机制解决Java中的线程安全问题
我们知道大多数程序都不会是单线程程序,单线程程序的功能非常有限,我们假设一下所有的程序都是单线程程序,那么会带来怎样的结果呢?假如淘宝是单线程程序,一直都只能一个一个用户去访问,你要在网上买东西还得等 ...
- 【Java中的线程】java.lang.Thread 类分析
进程和线程 联想一下现实生活中的例子--烧开水,烧开水时是不是不需要在旁边守着,交给热水机完成,烧开水这段时间可以去干一点其他的事情,例如将衣服丢到洗衣机中洗衣服.这样开水烧完,衣服洗的也差不多了.这 ...
- 关于Java中的线程安全(线程同步)
java中的线程安全是什么: 就是线程同步的意思,就是当一个程序对一个线程安全的方法或者语句进行访问的时候,其他的不能再对他进行操作了,必须等到这次访问结束以后才能对这个线程安全的方法进行访问 什么叫 ...
- Java中的线程同步
Java 中的线程同步问题: 1. 线程同步: 对于访问同一份资源的多个线程之间, 来进行协调的这个东西. 2. 同步方法: 当某个对象调用了同步方法时, 该对象上的其它同步方法必须等待该同步方法执行 ...
- JAVA基础——Java 中必须了解的常用类
Java中必须了解的常用类 一.包装类 相信各位小伙伴们对基本数据类型都非常熟悉,例如 int.float.double.boolean.char 等.基本数据类型是不具备对象的特性的,比如基本类型不 ...
- Java中的线程Thread总结
首先来看一张图,下面这张图很清晰的说明了线程的状态与Thread中的各个方法之间的关系,很经典的! 在Java中创建线程有两种方法:使用Thread类和使用Runnable接口. 要注意的是Threa ...
随机推荐
- SQL Server 本地时间和UTC时间的相互转换
SET @UTCDate = DATEADD(hour, DATEDIFF(hour,GETDATE(),GETUTCDATE()), @LocalDate) SET @LocalDate2 = DA ...
- C# 随机数 Radom 循环生成同一的数字
错误:在一个循环结构中,利用下列代码生成随机数,发生生成的随机数是一样的! for (int i = 0; i < myArray.Length; i++) //给数组赋值 { Random m ...
- 如何将一个用utf-8编码的文本用java程序转换成ANSI编码的文本
jdk有一个关于UTF-8的bug所以加了一句 br.skip(1); bugID: http://bugs.java.com/view_bug.do?bug_id=4508058 public st ...
- Reporting Service服务SharePoint集成模式安装配置(1、虚拟机+ 2、AD域环境配置)
研究 Reporting Service SharePoint 集成模式安装有一段时间,最初其实只是想知道Power View 技术是怎么回事,能实现什么效果.(当然也可以通过Excel 配置好 Po ...
- 【转】ANDROID自定义视图——onMeasure,MeasureSpec源码 流程 思路详解
原文地址:http://blog.csdn.net/a396901990/article/details/36475213 简介: 在自定义view的时候,其实很简单,只需要知道3步骤: 1.测量—— ...
- asp.netcore+jenkins+docker+svn+centos7.2 持续集成,每天凌晨获取最新代码打包发布
运行环境: centos7.2服务器或则虚拟机 可以是腾讯云也可以是内网服务器,(如果是内网服务器需要用frp做内网穿透,这样才可以通过外网访问该服务器) svnserver 来托管代码 一.安装je ...
- constexpr函数------c++ primer
constexpr函数是指能用于常量表达式的函数.定义constexpr函数的方法有其他函数类似,不过要遵循几项约定:函数的返回值类型及所以形参的类型都是字面值类型,而且函数体中必须有且只有一条ret ...
- React基础篇 (1)-- render&components
render 基础用法 //1.创建虚拟DOM元素对象 var vDom=<h1>hello wold!</h1> //2.渲染 ReactDOM.render(vDom,do ...
- 通过ssh访问NAT网络模式下的虚拟机Linux
链接:https://blog.csdn.net/jiuduan2009/article/details/51737004 https://blog.csdn.net/disalone201107/a ...
- 题解 P1720 【月落乌啼算钱】
题目链接 定义一个函数比较好求. #include<bits/stdc++.h>//万能头文件 using namespace std; double F(int x)//定义函数,为了保 ...