Concurrence Java - Interface de verrouillage
Une interface java.util.concurrent.locks.Lock est utilisée comme mécanisme de synchronisation des threads similaire aux blocs synchronisés. Le nouveau mécanisme de verrouillage est plus flexible et offre plus d'options qu'un bloc synchronisé. Les principales différences entre un verrou et un bloc synchronisé sont les suivantes:
Guarantee of sequence- Le bloc synchronisé ne fournit aucune garantie de la séquence dans laquelle le thread en attente aura accès. L'interface de verrouillage le gère.
No timeout- Le bloc synchronisé n'a pas d'option de temporisation si le verrouillage n'est pas accordé. L'interface de verrouillage offre une telle option.
Single method - Le bloc synchronisé doit être entièrement contenu dans une seule méthode alors que les méthodes lock () et unlock () d'une interface de verrouillage peuvent être appelées dans différentes méthodes.
Verrouiller les méthodes
Voici la liste des méthodes importantes disponibles dans la classe Lock.
N ° Sr. | Méthode et description |
---|---|
1 | public void lock() Acquiert la serrure. |
2 | public void lockInterruptibly() Acquiert le verrou sauf si le thread actuel est interrompu. |
3 | public Condition newCondition() Renvoie une nouvelle instance de Condition liée à cette instance de Lock. |
4 | public boolean tryLock() Acquiert le verrou uniquement s'il est libre au moment de l'appel. |
5 | public boolean tryLock() Acquiert le verrou uniquement s'il est libre au moment de l'appel. |
6 | public boolean tryLock(long time, TimeUnit unit) Acquiert le verrou s'il est libre dans le temps d'attente donné et que le thread en cours n'a pas été interrompu. |
sept | public void unlock() Libère le verrou. |
Exemple
Le programme TestThread suivant illustre certaines de ces méthodes de l'interface de verrouillage. Ici, nous avons utilisé lock () pour acquérir le verrou et unlock () pour libérer le verrou.
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
class PrintDemo {
private final Lock queueLock = new ReentrantLock();
public void print() {
queueLock.lock();
try {
Long duration = (long) (Math.random() * 10000);
System.out.println(Thread.currentThread().getName()
+ " Time Taken " + (duration / 1000) + " seconds.");
Thread.sleep(duration);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
System.out.printf(
"%s printed the document successfully.\n", Thread.currentThread().getName());
queueLock.unlock();
}
}
}
class ThreadDemo extends Thread {
PrintDemo printDemo;
ThreadDemo(String name, PrintDemo printDemo) {
super(name);
this.printDemo = printDemo;
}
@Override
public void run() {
System.out.printf(
"%s starts printing a document\n", Thread.currentThread().getName());
printDemo.print();
}
}
public class TestThread {
public static void main(String args[]) {
PrintDemo PD = new PrintDemo();
ThreadDemo t1 = new ThreadDemo("Thread - 1 ", PD);
ThreadDemo t2 = new ThreadDemo("Thread - 2 ", PD);
ThreadDemo t3 = new ThreadDemo("Thread - 3 ", PD);
ThreadDemo t4 = new ThreadDemo("Thread - 4 ", PD);
t1.start();
t2.start();
t3.start();
t4.start();
}
}
Cela produira le résultat suivant.
Production
Thread - 1 starts printing a document
Thread - 4 starts printing a document
Thread - 3 starts printing a document
Thread - 2 starts printing a document
Thread - 1 Time Taken 4 seconds.
Thread - 1 printed the document successfully.
Thread - 4 Time Taken 3 seconds.
Thread - 4 printed the document successfully.
Thread - 3 Time Taken 5 seconds.
Thread - 3 printed the document successfully.
Thread - 2 Time Taken 4 seconds.
Thread - 2 printed the document successfully.
Nous avons utilisé la classe ReentrantLock comme implémentation de l'interface Lock ici. La classe ReentrantLock permet à un thread de verrouiller une méthode même s'il a déjà le verrou sur une autre méthode.