Code Ready | Frontend
22.1K subscribers
1.17K photos
494 videos
17 files
852 links
Авторский канал по Frontend разработке.
Ресурсы, гайды, задачи, шпаргалки.
Информация ежедневно пополняется!

Автор: @energy_it

РКН: https://clck.ru/3NJCKs

Реклама на бирже: https://telega.in/c/code_ready
Download Telegram
Разбираем как не словить утечку памяти на обработчиках событий!

Одна из частых причин утечек — обработчик повесили, а снять забыли. Особенно это неприятно, когда событие висит не на локальном элементе, а на чём-то долгоживущем: window, document, глобальный event bus и т.п.
function mount() {
window.addEventListener('resize', onResize);
}

function unmount() {
// забыли снять обработчик
}


Компонента уже нет, а onResize всё ещё вызывается. Плюс callback может держать в замыкании старое состояние, DOM-ноды, данные — и GC не сможет это нормально прибрать.

Правильнее сразу закладывать cleanup:
function mount() {
window.addEventListener('resize', onResize);
}

function unmount() {
window.removeEventListener('resize', onResize);
}


Главное — снимать нужно ту же самую функцию.

Вот так не сработает:
el.addEventListener('click', () => doSomething());
el.removeEventListener('click', () => doSomething());


Это две разные функции, даже если выглядят одинаково.

Нормальный вариант:
function handler() {
doSomething();
}

el.addEventListener('click', handler);
el.removeEventListener('click', handler);


Ещё момент — capture.
el.addEventListener('click', handler, true);
el.removeEventListener('click', handler, false);


Так обработчик не снимется, потому что capture отличается. Для удаления браузер сравнивает type, listener и capture.
passive, once, signal в этом сравнении не участвуют.

Когда обработчиков несколько, удобно использовать AbortController:
const controller = new AbortController();

window.addEventListener('resize', onResize, {
signal: controller.signal
});

document.addEventListener('visibilitychange', onVisible, {
signal: controller.signal
});

// cleanup
controller.abort();


abort() снимет все обработчики, которые были добавлены с этим signal.

А если обработчик нужен только один раз:
button.addEventListener('click', handler, { once: true });


После первого вызова он удалится сам. Важный нюанс: удалённый DOM сам по себе ещё не утечка. Если на него нет внешних ссылок — GC его заберёт.

🔥 Проблемы начинаются, когда: обработчик висит на window / document; callback держит старое состояние в замыкании; DOM-нода или данные лежат в массиве, store или кэше.

📣 Code Ready | #практика
Please open Telegram to view this post
VIEW IN TELEGRAM
🔥148🤝7