Рина Сергеева
Рина Сергеева27 декабря 2018 г. 1:53

Немного об утечках памяти и способах их избежать

Что такое утечки памяти в Android-разработке?

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

Почему это происходит?

В Java есть собственные средства очистки памяти от неиспользуемых элементов. Это [ сборка мусора ] [id].

Сборщик мусора помечает все объекты, которые можно удалить, если на них нет ссылок. Утечка памяти - это просто потерянная ссылка, которая показывает, что объект не может быть удален.

Сложность этого бага в том, что до определенного времени он не виден и не может мешать.
Есть хорошая цитата Бенджамина Франклина: «Маленькая течь потопит большой корабль».
Утечки памяти занимают оперативную память приложения. Объем необработанной памяти будет расти, и однажды ваше приложение может замедлиться и вылететь . Это вызовет недовольство пользователей и, скорее всего, приложение будет удалено...


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

Как узнать, есть ли утечки памяти?

Вы можете найти память Лики по-разному. Но проще всего использовать Profiler Android-студию.

Порядок действий:

  • Стартовый проект
  • В нижней панели нажмите «Профилировщик».
  • Выберите «Память»
  • Нажать "Force Garbage Collection" (тогда нужно немного подождать)
  • Нажмите "Дамп кучи Java"
  • Фильтровать список по нужным классам
  • Посмотреть количество объектов

На рисунке выше показано, что существует четыре объекта Activity . Это произошло потому, что программист позволил программе потерять ссылки на представление . А пользователи любят поворачивать экран телефона. При каждой смене ориентации деятельность воссоздавалась заново, а старая оставалась лежать в глубине памяти.

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

Как избежать утечек памяти ?

Вот несколько советов:

1. Не создавайте статические ссылки на представления. Статические поля имеют тот же жизненный цикл, что и ваше приложение.

2. Не передавайте ссылки на объекты, которые живут дольше, чем объект, который вы передали. Например: не передавайте классу ссылки Runnable на View , так как новый поток будет продолжать live даже после повторного создания активности.

public class LeakingRunnable implements Runnable {

    private View view;

    LeakingRunnable(View view){       //don't do that!
        this.view = view;
    }
    @Override
    public void run() {
        // do some work
    }
}

И как перенести view в Runnable ?

Используйте другие типы ссылок.
В Java, помимо обычных "жестких ссылок" , есть и другие "мягкие ссылки" и "слабые" . Правильно их называть:

  • WeakReference
  • SoftReference
  • Phantom Reference

Наличие «мягких ссылок» больше не будет мешать сборщику мусора удалять Activity. Подробнее о них можно прочитать здесь: differences between weak, soft, phantom and normal links in Java

Без утечек памяти приведенный выше код выглядел бы так:

public class NoLeakingRunnable implements Runnable {

    @NonNull
    private final WeakReference<View> viewRef;

    public NoLeakingRunnable(@NonNull View view){
        this.viewRef = new WeakReference<>(view);         //do that!
    }
    @Override
    public void run() {
        View view = viewRef.get();
        // do some work
    }
}  

3. Необходимо сделать внутренние классы статическими Activity.

Класс внутренней активности о доступе к представлению (и любым другим объектам) создает синтетические ссылки на этот объект. И если внутренний класс живет дольше активности, то возникает утечка памяти.

В этой статье была рассказана лишь малая часть информации о том, что такое утечка памяти в Android-разработке и самые простые способы их избежать. Есть еще много сложных ситуаций. Некоторые из них будут рассмотрены в следующих статьях. Подпишитесь на новые статьи.

Рекомендуем хостинг TIMEWEB
Рекомендуем хостинг TIMEWEB
Стабильный хостинг, на котором располагается социальная сеть EVILEG. Для проектов на Django рекомендуем VDS хостинг.

Вам это нравится? Поделитесь в социальных сетях!

Комментарии

Только авторизованные пользователи могут публиковать комментарии.
Пожалуйста, авторизуйтесь или зарегистрируйтесь
Дмитрий

C++ - Тест 004. Указатели, Массивы и Циклы

  • Результат:60баллов,
  • Очки рейтинга-1
Дмитрий

C++ - Тест 003. Условия и циклы

  • Результат:92баллов,
  • Очки рейтинга8
d
  • dsfs
  • 26 апреля 2024 г. 11:56

C++ - Тест 004. Указатели, Массивы и Циклы

  • Результат:80баллов,
  • Очки рейтинга4
Последние комментарии
k
kmssr9 февраля 2024 г. 2:43
Qt Linux - Урок 001. Автозапуск Qt приложения под Linux как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
АК
Анатолий Кононенко5 февраля 2024 г. 9:50
Qt WinAPI - Урок 007. Работаем с ICMP Ping в Qt Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
EVA
EVA25 декабря 2023 г. 18:30
Boost - статическая линковка в CMake проекте под Windows Ошибка LNK1104 часто возникает, когда компоновщик не может найти или открыть файл библиотеки. В вашем случае, это файл libboost_locale-vc142-mt-gd-x64-1_74.lib из библиотеки Boost для C+…
J
JonnyJo25 декабря 2023 г. 16:38
Boost - статическая линковка в CMake проекте под Windows Сделал всё по-как у вас, но выдаёт ошибку [build] LINK : fatal error LNK1104: не удается открыть файл "libboost_locale-vc142-mt-gd-x64-1_74.lib" Хоть убей, не могу понять в чём дел…
G
Gvozdik19 декабря 2023 г. 5:01
Qt/C++ - Урок 056. Подключение библиотеки Boost в Qt для компиляторов MinGW и MSVC Для решения твой проблемы добавь в файл .pro строчку "LIBS += -lws2_32" она решит проблему , лично мне помогло.
Сейчас обсуждают на форуме
G
George137 мая 2024 г. 7:27
добавить qlineseries в функции в функции: "GPlotter::addSeries(QString title, QVector &arr)" я вызываю метод setChart(...), я в конструктор передал адрес на QChartView элемент
BlinCT
BlinCT5 мая 2024 г. 12:46
Написать свой GraphsView Всем привет. В Qt есть давольно старый обьект дял работы с графиками ChartsView и есть в 6.7 новый но очень сырой и со слабым функционалом GraphsView. По этой причине я хочу написать х…
PS
Peter Son4 мая 2024 г. 0:57
Best Indian Food Restaurant In Cincinnati OH Ready to embark on a gastronomic journey like no other? Join us at App india restaurant and discover why we're renowned as the Best Indian Food Restaurant In Cincinnati OH . Whether y…
Evgenii Legotckoi
Evgenii Legotckoi2 мая 2024 г. 21:07
Мобильное приложение на C++Qt и бэкенд к нему на Django Rest Framework Добрый день. По моему мнению - да, но то, что будет касаться вызовов к функционалу Андроида, может создать огромные трудности.
IscanderChe
IscanderChe30 апреля 2024 г. 11:22
Во Flask рендер шаблона не передаётся в браузер Доброе утро! Имеется вот такой шаблон: <!doctype html><html> <head> <title>{{ title }}</title> <link rel="stylesheet" href="{{ url_…

Следите за нами в социальных сетях