blob: d8811f4bff2b649539bd63e4ab173c0cb4ec39ce (
plain) (
blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
|
package com.yahoo.concurrent.classlock;
import java.util.HashMap;
import java.util.Map;
import java.util.function.BooleanSupplier;
/**
* @author valerijf
*/
public class ClassLocking {
private final Map<Class<?>, ClassLock> classLocks = new HashMap<>();
private final Object monitor = new Object();
public ClassLock lock(Class<?> clazz) {
return lockWhile(clazz, () -> true);
}
public ClassLock lockWhile(Class<?> clazz, BooleanSupplier interruptCondition) {
synchronized (monitor) {
while (classLocks.containsKey(clazz)) {
try {
monitor.wait();
} catch (InterruptedException ignored) {
}
}
if (!interruptCondition.getAsBoolean()) {
throw new LockInterruptException();
}
ClassLock classLock = new ClassLock(this, clazz);
classLocks.put(clazz, classLock);
return classLock;
}
}
void unlock(Class<?> clazz, ClassLock classLock) {
synchronized (monitor) {
if (classLock.equals(classLocks.get(clazz))) {
classLocks.remove(clazz);
monitor.notifyAll();
} else {
throw new IllegalArgumentException("Lock has already been released");
}
}
}
public void interrupt() {
synchronized (monitor) {
monitor.notifyAll();
}
}
}
|