Форум программистов, компьютерный форум, киберфорум
Java SE (J2SE)
Войти
Регистрация
Восстановить пароль
 
Рейтинг 4.86/7: Рейтинг темы: голосов - 7, средняя оценка - 4.86
29 / 29 / 3
Регистрация: 25.05.2014
Сообщений: 1,063
1

Почему Object.sleep() должен быть обязательно в synchronized?

30.03.2017, 22:47. Показов 1421. Ответов 4
Метки нет (Все метки)

Java
1
2
3
4
5
6
7
  synchronized (Thread.currentThread()){
            try {
                Thread.currentThread().wait();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
Почему вызов Object.wait() должен быть обязательно синхронизирован? Если убрать оператор synchronized, то получаю ошибку java.lang.IllegalMonitorStateException
Прощу прощения за ошибку в названии темы - должно быть wait() вместо sleep

Добавлено через 16 минут
synchronized/wait()/notify()/notifyAll() крепко увязаны вместе в концепции монитора (monitor). Это приводит к интересным побочным эффектам - нельзя вызывать у объекта wait()/notify()/notifyAll() вне блока синхронизации (или синхронизированного метода) по этому объекту (иначе будет IllegalMonitorStateException)
Вот что мне удалось найти - получается из-за того, что все завязано на концепции монитора?
0
Лучшие ответы (1)
Programming
Эксперт
94731 / 64177 / 26122
Регистрация: 12.04.2006
Сообщений: 116,782
30.03.2017, 22:47
Ответы с готовыми решениями:

Когда оператор должен быть обязательно членом класса, и когда он обязательно не принадлежит классу?
Нужно узнать случаи, когда оператор функция должна быть обязательно членом класса и случаи, когда...

Метод должен быть обязательно возвращающим
Программа работает корректно.Только с выводом не знаю,как сделать. class Program { ...

Где обязательно должен быть зарегистрирован сайт?
Добрый день! Раскручиваю проект по ВЧ, зарегил в каталогах постепенно, вопрос куда можно...

Должен ли быть обязательно установлен IE для работы WebBroser?
Люди подскажите, WebBroser - это просто ActiveX или всётаки приложение дерёт его из установленного...

__________________

Записывайтесь на профессиональные курсы Java-разработчиков
Обучение в Java Mentor с оплатой после трудоустройства
4
244 / 164 / 133
Регистрация: 30.09.2012
Сообщений: 690
30.03.2017, 23:42 2
Лучший ответ Сообщение было отмечено danek130995 как решение

Решение

Суть метода wait в том, чтобы освободить занятый монитор и перевести текущий поток в режим ожидания. Вне блока synchronized освобождать будет нечего, так как никакой монитор и не был занят. Отсюда и такие требования

Не по теме:

Возможно, я где-то не прав. Если что, поправьте :)

1
29 / 29 / 3
Регистрация: 25.05.2014
Сообщений: 1,063
30.03.2017, 23:58  [ТС] 3
Gr1f0nn, а я и не знал, что wait() освобождает монитор. А если монитор освобождается при wait(), разве это не будет означать, что объект перестанет быть синхронизирован? Ведь мониторы они разве не должны наоборот следить за тем, чтобы код был синхронизирован?
Получается в режиме ожидания монитор свободен? Я думал наоборот, не понимаю тогда
0
244 / 164 / 133
Регистрация: 30.09.2012
Сообщений: 690
31.03.2017, 00:17 4
danek130995, Представьте себе ситуацию, когда 2 потока работают с каким-то 1 объектом X. 1ый поток забирает что-то из X, а второй кладет что-то в X. Объект X имеет синхронизированные методы.

Java
1
2
3
4
5
6
7
8
9
10
11
class X {
    private int storage = 10;
    public synchronized int get() {
        storage--;
        return 1;
    }
 
    public synchronized  void put(int y) {
        storage += y;
    }
}
При такой реализации, может произойти ситуация, когда storage будет отрицательным.
Также одновременно что-то делать с объектом может только 1 поток, так как в качестве монитора выступает сам объект, но пытаться обращаться к объекту может N > 1 потоков.


Теперь, мы хотим сделать так, чтобы методом get можно было воспользоваться только тогда, когда storage > 0:

Java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
class X {
    private int storage = 10;
    public synchronized int get() {
        while(storage == 0) {
            wait();
        }
        storage--;
        return 1;
    }
 
    public synchronized  void put(int y) {
        storage += y;
        notify();
    }
}
Теперь, когда storage == 0, то поток, занявший монитор, освободит его до тех пор, пока storage не станет > 0 и какой-то другой поток не вызовет метод notify или notifyAll.
Если по шагам, то:
1. Поток, который занял монитор, вызвав метод get, освобождает его, вызовом метода wait;
2. 2й поток пытается занять освободившийся монитор, вызывая метод put;
3. В методе put вызывается метод notify текущего объекта
4. При завершении метода put 2й поток также освобождает монитор объекта

Отвечая на этот вопрос:
Цитата Сообщение от danek130995 Посмотреть сообщение
разве это не будет означать, что объект перестанет быть синхронизирован
Нет, не перестает. Объект либо потокобезопасен, либо нет.

Цитата Сообщение от danek130995 Посмотреть сообщение
Получается в режиме ожидания монитор свободен?
Не совсем так. Монитор может быть и занят, но уже другим потоком.
В режиме ожидания монитор освобождает тот поток, который его изначально и занял. Это дает возможность другим потокам попытаться занять освободившийся монитор, изменить состояние объекта, вызвать метод notify.

Возможно, где-то ошибся, но надеюсь, что мысль донес
1
29 / 29 / 3
Регистрация: 25.05.2014
Сообщений: 1,063
31.03.2017, 08:17  [ТС] 5
Gr1f0nn, спасибо! Я кажется понял - просто использование wait() уже подразумевает в себе то, что мы находимся в synchronized, иначе, как вы сказали, действительно, освобождать будет нечего. Поэтому wait() и заключается в блок synchronized.
0
IT_Exp
Эксперт
87844 / 49110 / 22898
Регистрация: 17.06.2006
Сообщений: 92,604
31.03.2017, 08:17

Заказываю контрольные, курсовые, дипломные работы и диссертации здесь.

Должен ли обязательно быть "человечек" в майнкрафт в режиме креатив (творчество)
Решила изучать питон с майнкрафт. И там и там - новичок. Изучаю по книжке, все делаю по инструкции....

Почему вызывается переопределённый метод Object.equals, хотя должен вызываться ПЕРЕГРУЖЕННЫЙ?
Суть: необходимо создать авторский класс, перегрузить в нём метод equals, создать объект этого...

Почему при наследовании QObject должен быть первым?
Если:class myItem:public QGraphicsRectItem,public QObject-не работает. Если:class myItem:public...

Почему столбец должен быть обязан использованым при обновлении
Здравствуйте форумчане. Делаю учебный проект. Столкнулся со странной проблемой. В пакедже...

Почему размер символьного массива должен быть больше на 1 символ, а не на 2?
Привет! Пока читал учебник Шилдта, возник вопрос: Почему Герберт Шилдт говорит, что следует...

Как должна быть организована сеть офиса? Что должно быть настроено обязательно, а чего быть не должно?
Добрый день. Сразу оговорюсь, что я инженер-строитель и настройкой сети занимаюсь только потому,...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
5
Ответ Создать тему
Опции темы

КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2021, vBulletin Solutions, Inc.