Semafori in Java
Utilizzando i costrutti primitivi di Java si possono realizzare
meccanismi più sofisticati (semafori e anche monitor).
public class Semaphore {
private int value;
public Semaphore (int initial){
value = initial;
}
synchronized public void signal()
++value;
notify();
}
synchronized public void wait()
throws InterruptedException {
while (value== 0) wait();
--value;
}
}
I metodi wait e signal sono a volte chiamati down e up.
La programmazione concorrente in Java - 1
Produttore/Consumatore con Semafori Java
class SemaBuffer implements Buffer {
protected
protected
protected
protected
protected
Object[] buf;
int in = 0;
int out = 0;
int count = 0;
int size = 0;
Semaphore full; //counts number of items
Semaphore empty; //counts number of spaces
SemaBuffer(int size) {
this.size=size; buf=new Object[size];
full = new Semaphore(0);
empty= new Semaphore(size);
}}
synchronized public void put(Object o)
throws InterruptedException {
empty.wait();
buf[in] = o; ++count; in=(in+1)%size;
full.signal();
}
synchronized public Object get()
throws InterruptedException{
full.wait();
Object o =buf[out]; buf[out]=null;
--count; out=(out+1)%size;
empty.signal();
return (o);
}
Funziona tutto bene???
La programmazione concorrente in Java - 2
Produttore/Consumatore con Semafori Java
L’esempio sopra non garantisce liveness, perché ci può
essere deadlock:
•
•
•
•
se il buffer è vuoto
consumatore fa una get
consumatore si blocca e rilascia il lock sul semaforo
produttore invoca put ma viene sospeso sul lock del
buffer, che non è stato rilasciato dal consumatore.
In questo caso il deadlock è causato da chiamate innestate
(nested monitor).
La programmazione concorrente in Java - 3
Esempio corretto di Produttore/Consumatore
con semafori Java
public void put(Object o)
throws InterruptedException {
empty.wait();
synchronized(this){
buf[in] = o; ++count; in=(in+1)%size;
}
full.signal();
}
Il deadlock può essere evitato solo attraverso un progetto
attento, sincronizzando la regione critica di accesso al buffer
solo dopo il passaggio del semaforo
Quindi la realizzazione di costrutti di sincronizzazione di
alto livello in Java richiede particolare attenzione.
Si possono anche realizzare variabili condizioni e monitor
come nella definizione di Hoare.
La programmazione concorrente in Java - 4