Android: явная утечка памяти при отправке большого количества сообщений через обработчик

У меня есть приложение, в котором поток пытается отправить множество объектов Location через обработчик в поток пользовательского интерфейса, где они анализируются и передаются другим потокам. Я заметил, что в приложении есть утечка памяти, так как оно вылетает через 2-3 часа после запуска. Если я использую DDMS для профилирования кучи, я замечаю постепенное увеличение количества «Объектов данных» (около 1500 в минуту). Похоже, что эти 1500 примерно поровну разделены между 16-байтовыми и 32-байтовыми элементами.

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

Я постепенно сузил необходимые части моей темы до этого:

class UIActivity implements Handler.Callback
{
@Override 
public void onCreate()
{
  m_handler = new Handler(this);
}

@Override
public boolean handleMessage(Message m)
{
  switch(m.what)
  { 
    case FAKE_LOCATION:
      m.obj = null; // desperate anti-leak measures
      return true;
  }
  return false;
}
public static class FakeLocationGenerator extends Thread
{
  private Handler m_callback;
  public FakeLocationGenerator(Handler callback)
  {
    m_callback = callback;
    start();
  }

  @Override
  public void run()
  {
    while(true)
    {
      Thread.sleep(50); // makes for 20hz
      Location l = new Location("fake");
      Message m = new Message();
      m.what = FAKE_LOCATION;
      m.obj = l;
      m_callback.SendMessage(m);
    }
  }
}
}

Реализация Handler.Callback.handleMessage() в действии пользовательского интерфейса просто возвращает значение true и ничего не делает с сообщением. Если я закомментирую вызов m_callback.SendMessage, то утечка памяти исчезнет. Я попробовал гигантский хак вызова Handler.removeMessages(FAKE_LOCATION) после каждого сообщения FAKE_LOCATION, но это ничего не дало.

Любые идеи?


person ArtHare    schedule 19.01.2012    source источник


Ответы (1)


Во-первых, вы должны использовать Message.obtain() и Message.recycle().

Во-вторых, возможно ли, что утечки нет, и вы просто генерируете новые сообщения быстрее, чем они обрабатываются...?

person Reuben Scratton    schedule 19.01.2012
comment
Ах, это может помочь. Когда я должен вызывать recycle() в этом случае? Еще один вопрос, который я прочитал во время исследования моей проблемы, указал, что recycle() автоматически вызывается после handleMessage(), и сказал, что мне не следует его вызывать. Я не думаю, что генерирую их слишком быстро (поскольку урезанный handleMessage() в настоящее время ничего не делает), но сегодня вечером я проверю, есть ли соотношение вызовов handleMessage() и sendMessage() 1:1. Вот ссылка, которую я нашел на recycle(): groups. google.com/group/android-developers/browse_thread/thread/ - person ArtHare; 19.01.2012
comment
Похоже, это сильно помогло. Я все еще сливаю информацию различными способами, но уже не через обработчика. - person ArtHare; 20.01.2012
comment
@ArtHare Итак, мы должны вызвать message.recycle? - person Kimi Chiu; 06.02.2017
comment
@KimiChiu У меня не очень хорошая память, но, похоже, я увеличил ответ, поэтому я думаю, что recycle () был ответом. - person ArtHare; 06.02.2017