Ответ 1
Нет, вам не нужно удалять $$watchers
, так как они будут эффективно удалены после уничтожения области.
Из Angular исходного кода (v1.2.21), Scope
$destroy
метод:
$destroy: function() {
...
if (parent.$$childHead == this) parent.$$childHead = this.$$nextSibling;
if (parent.$$childTail == this) parent.$$childTail = this.$$prevSibling;
if (this.$$prevSibling) this.$$prevSibling.$$nextSibling = this.$$nextSibling;
if (this.$$nextSibling) this.$$nextSibling.$$prevSibling = this.$$prevSibling;
...
this.$$watchers = this.$$asyncQueue = this.$$postDigestQueue = [];
...
Итак, массив $$watchers
опущен (и область удалена из иерархии области).
Извлечение watcher
из массива - это все незарегистрированная функция:
$watch: function(watchExp, listener, objectEquality) {
...
return function deregisterWatch() {
arrayRemove(array, watcher);
lastDirtyWatch = null;
};
}
Итак, нет смысла отменить регистрацию $$watchers
"вручную".
Вы все равно должны отменить регистрацию слушателей событий (как вы правильно отметили в своем сообщении)!
Примечание:
Вам нужно только отменить регистрацию слушателей, зарегистрированных в других областях. Нет необходимости отменить регистрацию слушателей, зарегистрированных в области, которая уничтожается.
Например:.
// You MUST unregister these
$rootScope.$on(...);
$scope.$parent.$on(...);
// You DON'T HAVE to unregister this
$scope.$on(...)
(спасибо to @John для указав это)
Кроме того, убедитесь, что вы отменили регистрацию каких-либо прослушивателей событий из элементов, которые переживают разрушаемую область. Например. если у вас есть директива, зарегистрируйте слушателя на родительском node или на <body>
, то вы также должны отменить регистрацию.
Опять же, вам не нужно удалять прослушиватель, зарегистрированный на уничтожаемом элементе.
Вид не связанный с исходным вопросом, но теперь есть также событие $destroyed
, отправленное на уничтожаемый элемент, поэтому вы также можете подключиться к нему (если это подходит для вашей usecase):
link: function postLink(scope, elem) {
doStuff();
elem.on('$destroy', cleanUp);
}