[Java并发包学习六]Semaphore介绍
2016-07-02 11:16
405 查看
概述
emaphore字面意思是信号量。他主要用于控制有限的资源的访问数量。我们看一个生活中常常出现的场景:一个厕所只有3个坑位,但是有10个人来上厕所,那怎么办?假设10的人的编号分别为1-10,并且1号先到厕所,10号最后到厕所。那么1-3号来的时候必然有可用坑位,顺利如厕,4号来的时候需要看看前面3人是否有人出来了,如果有人出来,进去,否则等待。同样的道理,4-10号也需要等待正在上厕所的人出来后才能进去,并且谁先进去这得看等待的人是否有素质,是否能遵守先来先上的规则。
在这种场景下,Semaphore便派上了用场。
基本函数介绍
构造函数
构造函数使用一个初始非零整数初始化Semaphore,表示最开始的时候的可用资源(上例中的厕所坑位)数量。有两种类型的构造函数,函数签名分别如下:1 2 | public Semaphore(int permits) public Semaphore(int permits, boolean fair) |
acquire函数
acquire函数用来申请资源,有两种不同的函数,签名分别如下:1 2 | public void acquire() throws InterruptedException public void acquire(int permits) throws InterruptedException |
acquireUninterruptibly函数
翻译过来就是不可能被打断的申请,对,该函数用来申请可用资源,并且不会被打断,也有两种类型,函数签名分别如下:1 2 | public void acquireUninterruptibly() public void acquireUninterruptibly(int permits) |
tryAcquire函数
tryAcquire函数用来获取可用资源,但是这类函数能够有时间的限制,如果超时,立即返回,有几种类型的函数,签名分别如下:1 23 | public boolean tryAcquire() public boolean tryAcquire(long timeout, TimeUnit unit) throws InterruptedException public boolean tryAcquire(int permits) public boolean tryAcquire(int permits, long timeout, TimeUnit unit) throws InterruptedException |
第二个函数用来申请一个资源,指定一个超时时间,如果当前可以资源数量足够,立即返回true,否则最多等待给定的时间,如果时间到还是未能获取资源,则返回false;如果等待过程中线程被打断,抛出InterruptedException异常;
和1一样,只是申请permits个资源;
和2一样,只是申请permits个资源。
release函数
release函数用来释放资源,就好比说蹲厕所的人出来了,让出了坑位。有两种类型,函数签名分别如下:1 2 | public void release() public void release(int permits) |
availablePermits函数
availablePermits函数用来获取当前可用的资源数量,函数签名如下:1 | public int availablePermits() |
drainPermits函数
drainPermits函数用来申请当前所有可用的资源,函数签名如下:1 | public int drainPermits() |
reducePermits函数
reducePermits函数用来禁止某些资源不可用,函数签名如下:1 | protected void reducePermits(int reduction) |
isFair函数
isFair函数用来判断当前的信号量是采用什么类型的策略,函数签名如下:1 | public boolean isFair() |
hasQueuedThreads函数
hasQueuedThreads函数用来判断是否有现成正在等待申请资源,函数签名如下:1 | public final boolean hasQueuedThreads() |
getQueueLength函数
getQueueLength函数用来返回正在等待申请资源的线程的数量,函数签名如下:1 | public final int getQueueLength() |
getQueuedThreads函数
getQueuedThreads函数返回当前正在等待申请资源的线程集合,函数签名如下:1 | protected Collection<Thread> getQueuedThreads() |
模拟实验
下面用代码来模拟如下场景:一个厕所只有3个坑位,有6个人(编号1-6)先后来上厕所,采用先来先上的策略。
代码如下:
1 23 | package com.winwill.test; import java.util.Random; import java.util.concurrent.Semaphore; /** * @author qifuguang * @date 15/8/25 23:23 */ public class TestSemaphore { public static void main(String[] args) throws Exception { Semaphore wc = new Semaphore(3, true); // 3个坑位 for (int i = 1; i <= 6; i++) { Thread t = new Thread(new Person("第" + i + "个人", wc)); t.start(); Thread.sleep(new Random().nextInt(300)); } } static class Person implements Runnable { private String name; private Semaphore wc; public Person(String name, Semaphore wc) { this.name = name; this.wc = wc; } public void run() { System.out.print(name + ":憋死老子了!"); if (wc.availablePermits() > 0) { System.out.println("天助我也,有坑位!"); } else { System.out.println("卧槽,没坑位了,等会儿吧..."); } try { wc.acquire(); //申请坑位 } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(name + ":终于轮到我了,拉屎就是爽!"); try { Thread.sleep(new Random().nextInt(1000)); // 模拟上厕所时间。 } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(name + ":拉完了,好臭!"); wc.release(); } } } |
第1个人:憋死老子了!天助我也,有坑位!
第1个人:终于轮到我了,拉屎就是爽!
第2个人:憋死老子了!天助我也,有坑位!
第2个人:终于轮到我了,拉屎就是爽!
第3个人:憋死老子了!天助我也,有坑位!
第3个人:终于轮到我了,拉屎就是爽!
第3个人:拉完了,好臭!
第4个人:憋死老子了!天助我也,有坑位!
第4个人:终于轮到我了,拉屎就是爽!
第5个人:憋死老子了!卧槽,没坑位了,等会儿吧…
第6个人:憋死老子了!卧槽,没坑位了,等会儿吧…
第2个人:拉完了,好臭!
第5个人:终于轮到我了,拉屎就是爽!
第4个人:拉完了,好臭!
第6个人:终于轮到我了,拉屎就是爽!
第1个人:拉完了,好臭!
第6个人:拉完了,好臭!
第5个人:拉完了,好臭!
可以看到,任何时候在用坑位的人数都控制在3人之内,并且开始蹲坑的顺序一定是按编号从小到大来的(公平策略)。
注意事项
release函数和acquire并没有要求一定是同一个线程都调用,可以A线程申请资源,B线程释放资源;调用release函数之前并没有要求一定要先调用acquire函数。
相关文章推荐
- Some elementary algorithms on discrete differential geometry(DDGSpring2016 Demos)
- [Java并发包学习五]CountDownLatch和CyclicBarrier介绍
- Jvm(jdk8)源码分析1-java命令启动流程详解
- java “值传递和引用传递”个人理解
- [Java并发包学习四]Future和FutureTask
- spring+EnCache缓存示例
- JAVA笔记-多态的意义(向上转型与向下转型)
- Java -AOP实现原理-动态代理的实现
- eclipse导入项目出现中文乱码解决方案
- Java基础——序列化
- Java千百问_08JDK详解(015)_JVMTI提供哪些功能
- java.util.concurrent并发包诸类概览
- Java中的内部类
- Timer类--计时器
- [Java并发包学习三]ThreadFactory介绍
- Java超级大火锅
- [Java并发包学习二]Executors介绍
- [javaEE] jsp的九大隐式对象
- 配置spring的web.xml
- JAVA之旅(二十三)——System,RunTime,Date,Calendar,Math的数学运算