Рина Сергеева
27 грудня 2018 р. 12:53

Трохи про витоки пам’яті та способи їх уникнення

Що таке витоку пам'яті в Android-розробці?

Програма створює об'єкти, вони лежать у пам'яті і не можуть їх очистити після завершення своєї роботи.

Чому це відбувається?

У Java є власні засоби очищення пам'яті від елементів, що не використовуються. Це складання сміття .

Складальник сміття позначає всі об'єкти, які можна видалити, якщо на них немає посилань. Виток пам'яті - це просто втрачене посилання, яке показує, що об'єкт не може бути видалений.

Складність цього бага у тому, що до певного часу він не видно і не може заважати.
Є хороша цитата Бенджаміна Франкліна: "Маленька текти потопить великий корабель".
Витоку пам'яті займають оперативну пам'ять програми. Об'єм необробленої пам'яті буде зростати, і одного разу ваш додаток може сповільнитися і вилетіти. Це викликає невдоволення користувачів і, швидше за все, додаток буде видалено.


Один із найнебезпечніших випадків, коли програма втрачає посилання на перегляд . Здавалося б, цей вигляд маленький на екрані. Однак варто пам'ятати, що подання має посилання на активність (фрагмент). А якщо посилання на перегляд не прибрати, то активність (фрагмент) теж жива. Активність (фрагмент) має посилання всі уявлення на екрані.

Як дізнатися, чи є витоку пам'яті?

Ви можете знайти пам'ять Ліки по-різному. Але найпростіше використовувати Profiler Android-студію.

Порядок дій:

  • Стартовий проект
  • У нижній панелі натисніть "Профілювальник".
  • Виберіть «Пам'ять»
  • Натиснути "Force Garbage Collection" (тоді потрібно трохи зачекати)
  • Натисніть "Дамп купи Java"
  • Фільтрувати список за потрібними класами
  • Подивитися кількість об'єктів

На малюнку вище показано, що є чотири об'єкти Activity . Це сталося тому, що програміст дозволив програмі втратити посилання на подання. Користувачі люблять повертати екран телефону. За кожної зміни орієнтації діяльність відтворювалася заново, а стара залишалася лежати у глибині пам'яті.

Якщо ви хочете, щоб спосіб вилову витоків пам'яті був більш цікавим, пропоную вам ознайомитися з бібліотекою LeakCanary .

Як уникнути витоків пам'яті ?

Ось кілька порад:

1. Не створюйте статичні посилання на уявлення. Статичні поля мають той же життєвий цикл, що й ваша програма.

** 2. Не передавайте посилання на об'єкти, які живуть довше, ніж об'єкт, який ви передали. повторного створення активності.

  1. public class LeakingRunnable implements Runnable {
  2.  
  3. private View view;
  4.  
  5. LeakingRunnable(View view){ //don't do that!
  6. this.view = view;
  7. }
  8. @Override
  9. public void run() {
  10. // do some work
  11. }
  12. }

І як перенести view у Runnable ?

Використовуйте інші типи посилань.
У Java, крім звичайних "жорстких посилань" , є й інші "м'які посилання" і "слабкі" . Правильно їх називати:

  • WeakReference
  • SoftReference
  • Phantom Reference

Наявність «м'яких посилань» більше не заважатиме збирачеві сміття видаляти Activity. Докладніше про них можна прочитати тут:

Без витоків пам'яті наведений вище код виглядав би так:

  1. public class NoLeakingRunnable implements Runnable {
  2.  
  3. @NonNull
  4. private final WeakReference<View> viewRef;
  5.  
  6. public NoLeakingRunnable(@NonNull View view){
  7. this.viewRef = new WeakReference<>(view); //do that!
  8. }
  9. @Override
  10. public void run() {
  11. View view = viewRef.get();
  12. // do some work
  13. }
  14. }

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

Клас внутрішньої активності доступу до подання (і будь-яким іншим об'єктам) створює синтетичні посилання цей об'єкт. І якщо внутрішній клас живе довше за активність, то виникає витік пам'яті.

У цій статті була розказана лише мала частина інформації про те, що таке витік пам'яті в Android-розробці та найпростіші способи їх уникнути. Є ще багато складних ситуацій. Деякі з них будуть розглянуті у таких статтях. Підпишіться нові статті.

По статті запитували0питання

2

Вам це подобається? Поділіться в соціальних мережах!

Коментарі

Only authorized users can post comments.
Please, Log in or Sign up
  • Останні коментарі
  • Evgenii Legotckoi
    16 квітня 2025 р. 17:08
    Благодарю за отзыв. И вам желаю всяческих успехов!
  • IscanderChe
    12 квітня 2025 р. 17:12
    Добрый день. Спасибо Вам за этот проект и отдельно за ответы на форуме, которые мне очень помогли в некоммерческих пет-проектах. Профессиональным программистом я так и не стал, но узнал мно…
  • AK
    01 квітня 2025 р. 11:41
    Добрый день. В данный момент работаю над проектом, где необходимо выводить звук из программы в определенное аудиоустройство (колонки, наушники, виртуальный кабель и т.д). Пишу на Qt5.12.12 поско…
  • Evgenii Legotckoi
    09 березня 2025 р. 21:02
    К сожалению, я этого подсказать не могу, поскольку у меня нет необходимости в обходе блокировок и т.д. Поэтому я и не задавался решением этой проблемы. Ну выглядит так, что вам действитель…
  • VP
    09 березня 2025 р. 16:14
    Здравствуйте! Я устанавливал Qt6 из исходников а также Qt Creator по отдельности. Все компоненты, связанные с разработкой для Android, установлены. Кроме одного... Когда пытаюсь скомпилиров…