Sémaphores
Un sémaphore compte le nombre de fois qu'une ressource limitée est en cours d'utilisation. Quand un composant a besoin d'une ressource partagée spécifique, il essaie de l'acquérir auprès du sémaphore qui la gère. Si la ressource n'est pas disponible, le fil d'exécution du demandeur est automatiquement bloqué. Dès qu'un autre composant libère la ressource, un des fils d'exécution en attente est réveillé et reprend le contrôle.
- public class Semaphore {
- private int count;
- public Semaphore(int n) {
- this.count = n;
- }
- public synchronized void acquire() {
- while (count == 0) {
- try {
- wait();
- }
- catch (InterruptedException e) {
- }
- }
- count--;
- }
- public synchronized void release() {
- count++;
- notify();
- }
- }
Une instance de Semaphore
est toujours initialisée avec le nombre total d'unités disponibles de la ressource qu'elle gère.
La méthode acquire
bloque le fil d'exécution tant que le compteur de la ressource est à zéro. Si la ressource est disponible, le compteur est décrémenté de un et l'exécution se poursuit normalement.
La méthode release
incrémente le compteur de la ressource de un et réveille tous les fils d'exécution en attente avec un appel à notify
. L'un d'eux reprendra la main, reviendra dans la boucle de la méthode acquire
et pourra continuer son exécution après avoir réservé la ressource.
Notez que acquire
et release
sont synchronisées afin de prévenir toute interférence sur le compteur.
Pour tester Semaphore
, nous allons créer une série de fils d'exécution qui essayeront tous et en même temps d'utiliser une ressource limitée pour un temps donné.
- public class SemaphoreTest {
- private Semaphore sema = new Semaphore(2);
- public void go() {
- for (int i = 0; i < 7; i++)
- (new Thread(new Consumer(i + 1, sema))).start();
- }
- private class Consumer implements Runnable {
- private Semaphore sema;
- private int id;
- Consumer(int id, Semaphore sema) {
- this.sema = sema;
- this.id = id;
- }
- public void run() {
- System.out.println(id + " - " + "Acquiring resource");
- sema.acquire();
- System.out.println(id + " - " + "Resource is locked");
- try {
- Thread.sleep((long) (Math.random() * 1000));
- }
- catch (InterruptedException e) {
- }
- System.out.println(id + " - " + "Releasing resource");
- sema.release();
- }
- }
- public static void main(String args[]) {
- (new SemaphoreTest()).go();
- }
- }
Le programme crée un Semaphore
qui autorise l'utilisation simultanée de deux unités d'une ressource, puis lance l'exécution en parallèle de sept instances de Consumer
qui ont toutes accès au sémaphore.
Un Consumer
affiche un message avant d'essayer d'acquérir la ressource, appelle acquire
, affiche un message dès que la ressource est disponible, fait quelque chose pendant un moment, affiche un dernier message avant de libérer la ressource et appelle release
.
Notez que le programme ne décide pas quand il peut prendre la ressource et continuer son exécution. Toute la logique est dans Semaphore
.
$ javac SemaphoreTest.java
$ java SemaphoreTest
1 - Acquiring resource
1 - Resource is locked
2 - Acquiring resource
2 - Resource is locked
3 - Acquiring resource
4 - Acquiring resource
...
2 - Releasing resource
3 - Resource is locked
3 - Releasing resource
4 - Resource is locked
4 - Releasing resource
1 - Releasing resource
5 - Resource is locked
...
L'ordre d'exécution des instances concurrentes de Consumer
peut varier.
Commentaires