Ошибка Blackberry IllegalStateException с использованием TimerTask - PullRequest
3 голосов
/ 20 октября 2010

У меня есть заставка, которая реализует MainScreen, и я пытаюсь использовать таймер, чтобы подождать 3 секунды, чтобы открыть новый экран.

TimerTask splashTask = new TimerTask() {
    public void run() {
        UiApplication.getUiApplication().pushScreen(new HomeNavigationScreen());
    }
};

timer.schedule(splashTask, 3000);

Однако, он выдает ошибку IllegalStateException, я не уверен, что я делаю неправильно?

Я новичок в затмении и Java, я не уверен, как просмотреть трассировку стека.

Вот что я нашел:

ApplicationManagerImpl.processExited : process process switching to background: pid=260
java.lang.IllegalStateException: UI engine accessed without holding the event lock.
Timer died: Thread[Thread-310688768,5]

Ответы [ 2 ]

6 голосов
/ 20 октября 2010

Диагностика на основе исходного вопроса:

Ваш первый шаг должен состоять в том, чтобы прочитать документацию для Timer и TimerTask, которая дает следующие условия для IllegalStateException при schedule(...):

Если поток выполнения задачи таймера неожиданно завершается, например, из-за вызова его метода остановки, любая дальнейшая попытка запланировать задачу на таймере приведет к исключению IllegalStateException, как если бы отмена таймераметод был вызван.

Вы когда-нибудь .stop()?Ты когда-нибудь звонил cancel()?Можете ли вы проверить, не вызвало ли задание исключение, которое внезапно завершило поток Timer?Вы используете тот же Timer для других TimerTask с?Вы им доверяете?

Попробуйте обернуть свою логику TimerTask широким обработчиком исключений и попросите ее напрямую отчитаться, чтобы увидеть, есть ли внутреннее непредвиденное исключение, которое приведет к сбою следующего вызова schedule какописано.

Кроме того, придирка, но важная: передовая практика, когда задаешь вопрос, который включает в себя «Почему я получил это исключение?»должен включать полное исключение и вывод трассировки стека.Вы получите гораздо более качественные ответы, и вам больше не придется читать этот текст, а затем придется заново редактировать свой вопрос и т. Д.

Руководство по вашей конкретной ошибке:

Хорошо, прочитайте заявление об исключении.Похоже, вы расстроили движок пользовательского интерфейса.Таким образом, у вас есть две проблемы:

  1. Вы расстроили поток таймера, передав ему задачу, которая разрывается с необработанным исключением.Хорошая политика - обернуть эти вещи в блок try/catch, где операция catch равна не менее , чтобы напечатать или записать трассировку используемого стека и повторно выдать исключение.

  2. Вы расстроили логику событий пользовательского интерфейса, см. Ниже ...

Я не знаю пользовательский интерфейс Blackberry, но, вероятно, то, что вы пытаетесь сделать, нужнобыть сделано в цикле событий GUI.Эта запись KB должна помочь.А еще лучше, прочитайте документацию API .Вам нужно удерживать блокировку графического интерфейса, как в Swing, для вызова pushScreen().Один из способов сделать это - изменить свой код для звонка через invokeLater() или invokeAndWait().

Код кандидата

Это не проверено, как я никогда и не делалЯ не планирую заниматься какой-либо разработкой BlackBerry, но, по-моему, она совпадает с опубликованным API-интерфейсом BlackBerry, FWIW.Попробуйте что-то вроде этого:

TimerTask splashTask = new TimerTask() 
{
  public void run() {
    final UiApplication uia = UiApplication.getUiApplication();
    final Object eventLock = uia.getEventLock();
    synchronized(eventLock) {
       uia.pushScreen(new HomeNavigationScreen());
    }
  }
};

timer.schedule(splashTask, 3000);

или, что менее вероятно, приведет к проблемам с синхронизацией и потенциальным взаимоблокировкам:

TimerTask splashTask = new TimerTask() 
{
  public void run() {
    final UiApplication uia = UiApplication.getUiApplication();
    uia.invokeLater(new Runnable() { 
      public void run() {
        uia.pushScreen(new HomeNavigationScreen());
     });
    }
  }
};

timer.schedule(splashTask, 3000);
2 голосов
/ 20 октября 2010

Как отметил Андерсой, вы пытаетесь внести изменения в пользовательский интерфейс, не удерживая блокировку событий пользовательского интерфейса.invokeLater - это механизм получения кода для выполнения и безопасного внесения изменений в пользовательский интерфейс.

TimerTask splashTask = new TimerTask() {
    public void run() {
        UiApplication.getUiApplication().invokeLater(new Runnable() {
            public void run() {
                UiApplication.getUiApplication().pushScreen(new HomeNavigationScreen());
            }
        }
    }
};

timer.schedule(splashTask, 3000);
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...