Semaphore 是一个控制访问多个共享资源的计数器。

当一个线程想要访问某个共享资源,首先,它必须获得 semaphore。如果 semaphore 的内部计数器的值大于 0,那么 semaphore 减少计数器的值并允许访问共享的资源。计数器的值大于 0 表示,有可以自由使用的资源,所以线程可以访问并使用它们。另一种情况,如果 semaphore 的计数器的值等于 0,那么 semaphore 让线程进入休眠状态一直到计数器大于 0。计数器的值等于 0 表示全部的共享资源都正被线程们使用,所以此线程想要访问就必须等到某个资源成为自由的。当线程使用完共享资源时,它必须释放 semaphore 以便其他线程可以访问共享资源。这个操作会增加 semaphore 的内部计数器的值。

二进制信号量

二进制信号量(binary semaphores)是一种比较特殊的信号量,这种信号量只保护访问唯一的共享资源,它的内部计数器值只能是 1 或者 0。

假设有若干个线程排队执行打印任务,打印机在同一时间只能执行单个任务,打印过程中其他线程只能挂起等待。

public class Printer {

    private final Semaphore semaphore;

    public Printer() {
semaphore = new Semaphore(1);
} public void print(Object doc) {
try {
semaphore.acquire();
Long duration = (long) (Math.random() * 10);
System.out.printf("%s: PrintQueue: Printing a Job during %d seconds\n", Thread.currentThread().getName(), duration);
TimeUnit.SECONDS.sleep(duration);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
semaphore.release();
}
} } public class PrintJob implements Runnable { private Printer printer; public PrintJob(Printer printer) {
this.printer = printer;
} @Override
public void run() {
System.out.printf("%s: Going to print a job\n", Thread.currentThread().getName());
printer.print(new Object());
System.out.printf("%s: The document has been printed\n", Thread.currentThread().getName());
} } public class Main { public static void main (String args[]){
Printer printer = new Printer(); Thread[] threads = new Thread[10];
for (int i = 0; i < 10; i++){
threads[i] = new Thread(new PrintJob(printer), "Thread" + i);
threads[i].start();
}
} }

运行结果:

Thread1: Going to print a job
Thread6: Going to print a job
Thread9: Going to print a job
Thread7: Going to print a job
Thread8: Going to print a job
Thread4: Going to print a job
Thread5: Going to print a job
Thread3: Going to print a job
Thread0: Going to print a job
Thread2: Going to print a job
Thread1: PrintQueue: Printing a Job during 7 seconds
Thread1: The document has been printed
Thread6: PrintQueue: Printing a Job during 5 seconds
Thread6: The document has been printed
Thread9: PrintQueue: Printing a Job during 4 seconds
Thread9: The document has been printed
Thread7: PrintQueue: Printing a Job during 8 seconds
Thread7: The document has been printed
Thread8: PrintQueue: Printing a Job during 6 seconds
Thread8: The document has been printed
Thread4: PrintQueue: Printing a Job during 2 seconds
Thread4: The document has been printed
Thread5: PrintQueue: Printing a Job during 0 seconds
Thread5: The document has been printed
Thread3: PrintQueue: Printing a Job during 7 seconds
Thread3: The document has been printed
Thread0: PrintQueue: Printing a Job during 0 seconds
Thread0: The document has been printed
Thread2: PrintQueue: Printing a Job during 5 seconds
Thread2: The document has been printed

这个例子的关键是 Printer 类的 print() 方法。此方法展示了当你使用 semaphore 来实现临界区以保护共享资源时必须遵守的三个步骤:

  1. 首先,调用 acquire() 方法获得信号量;
  2. 然后,操作共享资源;
  3. 最后,调用 release()  方法释放信号量。

另一个重点是 Printer 类的构造方法和初始化 Semaphore 对象。将信号量初始化为 1,就是创建了一个二进制信号量。二进制信号量只保护一个共享资源,这个例子中就是 printer。当开始 10 个线程时,第一个获得 semaphore 的线程就获得临界区的访问权,其他线程都会被 semaphore 阻塞,直到获得 semaphore 的线程释放它。当 semaphore 被释放时,它会在等待的线程中选择其中一个并赋予其临界区的访问权。所有的打印任务都会被执行,只是它们需要排队,一个接一个地执行。

控制并发访问多个共享资源

在上个例子,使用二进制信号量来保护一个共享资源。但是,信号量也可以用来保护多个共享资源。下面的类使用信号量控制对内容池的访问:

class Pool {
private static final int MAX_AVAILABLE = 100;
private final Semaphore available = new Semaphore(MAX_AVAILABLE, true); public Object getItem() throws InterruptedException {
available.acquire();
return getNextAvailableItem();
} public void putItem(Object x) {
if (markAsUnused(x))
available.release();
} // Not a particularly efficient data structure; just for demo protected Object[] items = ... whatever kinds of items being managed
protected boolean[] used = new boolean[MAX_AVAILABLE]; protected synchronized Object getNextAvailableItem() {
for (int i = 0; i < MAX_AVAILABLE; ++i) {
if (!used[i]) {
used[i] = true;
return items[i];
}
}
return null; // not reached
} protected synchronized boolean markAsUnused(Object item) {
for (int i = 0; i < MAX_AVAILABLE; ++i) {
if (item == items[i]) {
if (used[i]) {
used[i] = false;
return true;
} else
return false;
}
}
return false;
} }

公平模式

与 ReentrantLock 一样,Semaphore 也提供一个接收 fair 参数的构造方法。当此参数置为 true 时,Semaphore 保证等待时间最久的线程优先获得信号量。

Semaphore 的方法

Semaphore 类有另外 2 个版本的 acquire() 方法:

  • acquireUninterruptibly():acquire()方法是当 semaphore 的内部计数器的值为 0 时,阻塞线程直到 semaphore 被释放。在阻塞期间,线程可能会被中断,然后此方法抛出 InterruptedException 异常。而此版本的 acquire 方法会忽略线程的中断而且不会抛出任何异常。
  • tryAcquire():此方法会尝试获取 semaphore。如果成功,返回true。如果不成功,返回 false 值,并不会被阻塞和等待 semaphore 的释放。

acquire、acquireUninterruptibly、tryAcquire 和 release 方法有一个外加的包含一个 int 参数的版本。这个参数表示线程想要获取或者释放 semaphore 的许可数。

Java Concurrency - Semaphore 信号量的更多相关文章

  1. java多线程-Semaphore信号量使用

    介绍 信号量(Semaphore),有时被称为信号灯,是在多线程环境下使用的一种设施, 它负责协调各个线程, 以保证它们能够正确.合理的使用公共资源. 概念 Semaphore分为单值和多值两种,前者 ...

  2. Java中Semaphore(信号量)的使用

    Semaphore的作用: 在java中,使用了synchronized关键字和Lock锁实现了资源的并发访问控制,在同一时间只允许唯一了线程进入临界区访问资源(读锁除外),这样子控制的主要目的是为了 ...

  3. java多线程----Semaphore信号量

    import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util ...

  4. Java中Semaphore(信号量) 数据库连接池

    计数信号量用来控制同时访问某个特定资源的操作数或同时执行某个指定操作的数量 A counting semaphore.Conceptually, a semaphore maintains a set ...

  5. java中的信号量Semaphore

    Semaphore(信号量)充当了操作系统概念下的“信号量”.它提供了“临界区中可用资源信号量”的相同功能.以一个停车场运作为例.为了简单起见,假设停车场只有三个车位,一开始三个车位都是空的.这时如果 ...

  6. java笔记--对信号量Semaphore的理解与运用

    java Semaphore 信号量的使用: 在java中,提供了信号量Semaphore的支持. Semaphore类是一个计数信号量,必须由获取它的线程释放, 通常用于限制可以访问某些资源(物理或 ...

  7. Java并发编程笔记之Semaphore信号量源码分析

    JUC 中 Semaphore 的使用与原理分析,Semaphore 也是 Java 中的一个同步器,与 CountDownLatch 和 CycleBarrier 不同在于它内部的计数器是递增的,那 ...

  8. java并发之(4):Semaphore信号量、CounDownLatch计数锁存器和CyclicBarrier循环栅栏

    简介 java.util.concurrent包是Java 5的一个重大改进,java.util.concurrent包提供了多种线程间同步和通信的机制,比如Executors, Queues, Ti ...

  9. java架构之路(多线程)JUC并发编程之Semaphore信号量、CountDownLatch、CyclicBarrier栅栏、Executors线程池

    上期回顾: 上次博客我们主要说了我们juc并发包下面的ReetrantLock的一些简单使用和底层的原理,是如何实现公平锁.非公平锁的.内部的双向链表到底是什么意思,prev和next到底是什么,为什 ...

随机推荐

  1. ALAsset,ALAssetsLibrary,ALAssetsgroup常见属性及用法

    转载自  http://www.cnblogs.com/javawebsoa/archive/2013/07/19/3201246.html ALAssetsgroup --------------- ...

  2. POJ2001Shortest Prefixes(字典树)

    题目大意就是帮你给N条字符串,每条长度不超过20.问要将他们单一识别出来,每个字符串最短可以缩为多短. 如: abc abcdefg bc adef 这四个字符串就可分别缩写为 abc abcd b ...

  3. 12个有趣的C语言面试题

    摘要:12个C语言面试题,涉及指针.进程.运算.结构体.函数.内存,看看你能做出几个! 1.gets()函数 问:请找出下面代码里的问题: #include<stdio.h> int ma ...

  4. 在数据库各种状态下查询DBID的五大类十种方法汇总

    关于DBID: DBID是DataBase IDentifier的缩写,意思就是数据库的唯一标识符. 这个DBID在数据文件头和控制文件都是存在的,可以用于标示数据文件的归属. 对于不同数据库来说,D ...

  5. Cloud Computing Deployment Models

    Cloud computing can broadly be broken down into three main categories based on the deployment model. ...

  6. zendstudio 10下载汉化

      1.文件和汉化文件 ZendStudio官方下载地址:http://www.geekso.com/component/zendstudio-downloads/ 百度云地址: 10.0.0.msi ...

  7. Redis命令小细节

    1.  set   setnx   setex set  将字符串 value的值关联到key ,假设key已经存在,那么覆盖原来的,假设不存在.那么就创建 setnx  将key的值设置为value ...

  8. JAVA-开发环境搭建之JDK安装配置教程

    在进行java开发前先要搭建java的开发环境 下载java的开发环境eclipse 安装&配置环境变量 1,JDK安装

  9. Codeforces Round #328 (Div. 2) C. The Big Race 数学.lcm

    C. The Big Race Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/contest/592/probl ...

  10. 细说linux挂载——mount,及其他……

    http://forum.ubuntu.org.cn/viewtopic.php?t=257333