Здравствуйте, я узнаю о блокировке повторного входа и переменной условия в Java. Мне довелось встретить это tutorial. В учебнике, автор предоставляет производитель-потребитель пример использующего ReentrantLock:ReeantrantLock и переменная условия
public class CondDemo
{
public static void main(String[] args)
{
Shared s = new Shared();
new Producer(s).start();
new Consumer(s).start();
}
}
class Shared
{
private volatile char c;
private volatile boolean available;
private final Lock lock;
private final Condition condition;
Shared()
{
c = '\u0000';
available = false;
lock = new ReentrantLock();
condition = lock.newCondition();
}
Lock getLock()
{
return lock;
}
char getSharedChar()
{
lock.lock();
try
{
while (!available) {
try
{
condition.await();
}
catch (InterruptedException ie)
{
ie.printStackTrace();
}
}
available = false;
condition.signal();
}
finally
{
lock.unlock();
return c;
}
}
void setSharedChar(char c)
{
lock.lock();
try
{
while (available) {
try
{
condition.await();
}
catch (InterruptedException ie)
{
ie.printStackTrace();
}
}
this.c = c;
available = true;
condition.signal();
}
finally
{
lock.unlock();
}
}
}
class Producer extends Thread
{
private final Lock l;
private final Shared s;
Producer(Shared s)
{
this.s = s;
l = s.getLock();
}
@Override
public void run()
{
for (char ch = 'A'; ch <= 'Z'; ch++)
{
l.lock();
s.setSharedChar(ch);
System.out.println(ch + " produced by producer.");
l.unlock();
}
}
}
class Consumer extends Thread
{
private final Lock l;
private final Shared s;
Consumer(Shared s)
{
this.s = s;
l = s.getLock();
}
@Override
public void run()
{
char ch;
do
{
l.lock();
ch = s.getSharedChar();
System.out.println(ch + " consumed by consumer.");
l.unlock();
}
while (ch != 'Z');
}
}
являются методами l.lock()
и l.unlock()
в Producer
и Consumer
ненужных? Я замечаю, что мы уже применили блокировку/разблокировку в методах объекта getSharedChar()
и setSharedChar()
. Или это предложенная модель использования переменных условия?
Это не очень хорошая практика, можете ли вы предоставить ссылку для лучшего примера?
Спасибо :)
Хм, странно. Читая этот код, мне кажется, что заблокированная блокировка может быть заблокирована снова - разве это не должно исключать исключение или что-то еще? – Manu
Нет, это основная функция блокировки входа. Однако при получении блокировки есть [счет получения] (https://docs.oracle.com/javase/8/docs/api/java/util/concurrent/locks/ReentrantLock.html#getHoldCount--), чтобы обеспечить блокировку освобождение осуществляется в нужном месте. – LoganMzz
Но в чем преимущество получения нескольких замков? – hackjutsu